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ABSTRACT 

The Computer Aided Prototyping System (CAPS) is a systems engineering tool 
intended to make the iterative process of software development more efficient. The 
simplest way to input and modify a CAPS design is through the graphical editor. When a 
design is modified over and over, the resultant graphical representation can become 
difficult to comprehend. Trying to change the graphical representation by hand can be 
very tedious. 

By adding automatic layout techniques to the graphical editor, this task is made 
easier for the user of the system. Automatic layout techniques for general graphs that 
maximize all of the aesthetic characteristics of a graph are not possible. One 
characteristic may conflict with another. By giving the user multiple layout algorithms 
that emphasis different characteristics over others, the user may choose between different 
layouts for the graphical representation. 

Since CAPS was in the middle of a restructure and no graphical editor was 
available, automatic layout techniques were investigated using other graphical editors. 

Graphs with characteristics similar to a CAPS graph were input into the graphical editors 
and then the layout algorithms applied. The results of this assessment proved that the 
addition of automatic layout techniques to CAPS would improve performance. The 
library of layout algorithms will be incorporated into the new graphical editor in CAPS. 
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I. INTRODUCTION 


A. GENERAL 

The chore of programming a computer to perform a desired task has become 
increasingly difficult over the past few decades. The tasks that computer programmers 
are asked to complete are much more complex and difficult than previously attempted. 
However, the tools that computer programmers have utilized to perform these tasks have 
not enabled these systems to be easily built and maintained. 

The trend in the field of software engineering is towards automated tools that 
handle the more tedious components of the software engineering process. This allows the 
computer programmer to apply more resources to the creative process by trying different 
approaches without investing too many assets up front. 

The Computer Aided Prototyping System (CAPS) is an integrated set of tools that 
allows a software engineer to design large software systems and test their design prior to 
implementation. The software engineer can start designing a system with CAPS using a 
top down approach. At each level of detail, the system can be tested to ensure that the 
system’s performance is within the parameters of the desired end product. 

To ease the process, CAPS has a graphical interface for the design process. Each 
operator can be laid out in the graphical display. These operators are connected to each 
other with directed lines that represent streams. Each operator can be subdivided into 
simpler operators until all the operators are atomic operators. Each operator can have 
performance constraints associated with it. CAPS can test that these constraints are not 
violated by the overall design. If a violation occurs, the design can be modified until the 
desired system specifications are reached. 
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B. PROBLEM STATEMENT 


This research is a revision of the work on the graphical editor of CAPS. The 
current graphical editor has no automated layout capabilities. The program graph can 
become convoluted after many edits. Automated layout techniques will be investigated 
for use in the CAPS graphical editor that will minimize such attributes as crossing lines 
and spacing. 

By adding automatic layout techniques to the graphical editor, 
designer/programmer productivity is gained by helping them get the job done faster. 
Further, the resulting program graphs will be easier to comprehend. 


C. SCOPE 


The scope of this thesis will deal with the capabilities of layout techniques for 
directed graphs while simultaneously analyzing the unique needs of the graphical editor 
in CAPS. Once both are completed, findings will be used to build an automated layout 
function that will meet the needs of the CAPS in the most user fi'iendly means available. 

This thesis contains three primary products. The first is a survey of current 
automated layout algorithms for graphs. The second is an evaluation of the properties of 
a common program graph in CAPS. The third is the implementation of automated layout 
algorithms for the CAPS graphical editor. 
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11. BACKGROUND KNOWLEDGE 


A. GENERAL 

The Computer Aided Protot 3 ^ing System (CAPS) is a software-engineering tool 
designed to take the drudgery out of prototyping systems. A software developer can 
design a software system, test the design, and add timing requirements. Most projects 
become an iterative process of re-visiting the design, making the necessary changes to fix 
problems, and then testing the new design. 

After a few iterations, the graphical display of the prototyped system can become 
very difficult to understand, since the graphical display requires the user to layout the 
design. Fixing the design layout by hand requires some time. The layout editor in CAPS 
is also rather slow and plodding for the user. By providing automatic layout techniques 
to the user, the chore of laying out the design in a more coherent manner is lessened. 

B. GRAPH CHARACTERISTICS 

All graphs have different characteristics. By testing a graph for these 
characteristics, a graph can be categorized into different graph categories. These graph 
categories include trees, planar undirected graphs, planar directed graphs, general 
undirected graphs, general directed graphs, etc. 

Graphs can be trees if no cycles exist. They can be planar if the display of the 
graph is meant to be two-dimensional. 

A general graph can be displayed with any-dimensional perspective. The edges 
connecting the graphs can also differ in category. In general, directed graphs have an 
edge with at least one arrow signifying direction. Undirected graphs have no arrows and 
there is no restriction on the direction. Figure 2.1 shows several different graphs with 
differing characteristics. 
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A. Undirected, acyclic, 
planar graph 




B. Undirected, cyclic, 
planar graph 


D. Undirected, cyclic, 
general graph 


Figure 2.1: Different characteristics of graphs 


Another characteristic important in laying a graph out is whether polyline 


drawings are permitted. Polyline drawings allow for bends in an edge. Straight-line 


drawings only allow for straight-lines to be drawn between vertices. Figure 2.2 


demonstrates this problem. 



Figure 2.2: Polyline drawing vs Straight-line drawing 
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An orthogonal drawing maps each edge into a chain of horizontal and vertical 
segments. This produces a boxy looking graph. Figure 2.3 gives an example of an 
orthogonal graph. 



Figure 2.3: Orthogonal graph 


C. AUTOMATIC LAYOUT TECHNIQUES 

Automated layout techniques for a graph is an ongoing research area. Laying out 
a generic graph optimally is considered a.NP Complete task. However, there exist 
algorithms that can approximate an optimal solution. By having the computer do most of 
the tedious work, a user optimize the layout with minor adjustments. 

Graph layout techniques can be written to layout a generic graph or a subset of the 
set of all graphs. A tree graph would use an algorithm whose sole purpose is to optimize 
the layout of trees. This algorithm would not work for graphs that have cycles. Another 
class of algorithms would be used for graphs with cycles. 

An optimal presentation of a generic graph is based on aesthetic criteria. 

Aesthetic criteria attempt to characterize readability by means of general optimization 
goals. These goals consist of: 
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1. Minimize crossing 


2. Minimize area 


3. Minimize bends (in orthogonal drawings) 


4. Minimize slopes (in polyline drawings) 


5. Maximize smallest slope 


6. Maximize display of symmetries 


In general, one cannot simultaneously optimize two aesthetic criteria. A simple 
example of this problem can be shown in Figure 2.4. A simple graph of four nodes and 
six links is displayed with different criteria. Since these criteria cannot optimize all 
aesthetic criteria, a set of constraints is usually provided as additional input to a graph 
drawing algorithm. These constraints specify which criteria are more important for the 
desired layout. 
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A. Minimize crossing B. Maximize symmetries 

Figure 2.4: Differences in optimization with different criteria. 


By allowing a blending of different criteria the user can automatically change the 
appearance of the display by playing with the constraints. This leads to many complexity 
issues. Testing planarity takes linear time. Testing upward planarity is NP-hard. 
Minimizing crossing is NP-hard. All of these constraints add to the complexity of the 
algorithms. They also add to the time taken to compute a layout. 
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III. ANALYSIS OF AUTOMATED LAYOUT TECHNIQUES 


A. GENERAL 

Numerous algorithms exist for automatically laying out graphs. Since the optimal 
layout for a graph is a NP Complete problem, these algorithms are only approximations 
to an optimal layout. Almost every algorithm returns a different solution to the question 
for a specific graph. Some algorithms give different answers when called repeatedly. 
Usually, a library of algorithms are given to the user to interactively select a layout by 
running different algorithms until he finds a layout that is aesthetically pleasing and more 
coherent. 

B. GENERAL TECHNIQUES 

In general, all algorithmic approaches to laying out a graph in an aesthetically 
pleasing manner can be broken down into categories. Inside each category there are 
many variations of the general theme. This section will outline the general themes of 
these main categories. 

Trees or rooted trees are often used to represent hierarchies such as family trees, 
organizational charts, and search trees. Planar straight-line drawings and orthogonal 
polyline drawings are commonly used to represent rooted trees. In general, vertices are 
placed along horizontal lines according to their level. There is a minimum separation 
distance between two consecutive vertices on the same level. The width of the drawing is 
as small as possible. Also, for binary trees, left and right children of each vertex are 
placed to the left and right of the vertex, respectively. 

In an inclusion representation of a tree, boxes represent nodes and parent-child 
relationships are represented by inclusion of one box in another. The tip-over convention 
is similar to the classical tree graph, however, children of some nodes may be arranged 
vertically rather than horizontally. 
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Free trees do not represent hierarchies and have no specific root. Selecting at 
random a root and then applying an algorithm for a rooted tree works adequately. 

Straight-line drawing calls for minimizing crossing of lines in a general graph 
while minimizing the space required to display the graph and using only straight lines 
between edges, no polylines. The best approaches to this problem, to date, are heuristic 
based on a physical model. 

The spring embedder algorithm takes such an approach. The drawing process is 
to simulate a mechanical system, where vertices are replaced by rings, and edges are 
replaced by springs. The springs attract the rings if they are too far apart, and repel them 
if they are too close. Variations to this model modify the energy function of the springs 
by criteria other than just distance. 

Planarization involves ensuring that a general graph is planar. If not, it attempts 
to planarize the graph. This allows many techniques that have been developed for planar 
graphs to be used. 

The most common planarization operation is edge deletion. The smallest set of 
edges whose deletion yields a planar graph is found. This is equivalent to finding a 
planar subgraph with a large number of edges. Most of the algorithms that use edge 
deletion use different approaches for finding a maximum planar subgraph. 

Another technique for planarization is splitting. The splitting operation is to make 
two copies of a vertex and share the neighbors between the two copies. Algorithms that 
use splitting try to optimize the finding of a minimum splitting sequence. 


C. ALGORITHM SEARCH 

Relatively little information on graph layout algorithms is available. Only 
through a search on the Internet was the needed information found. There exist a few 
home pages on the Internet that deal with automatic layout algorithms. Books on the 
subject are just emerging. 
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D. THE GRAPH DRAWING SERVER (GDS) 


The graph drawing server is located via Dr. Tamassia’s home page at 
htt p://www.cs.brovvn.edu/Deople/rt . This server is a collection of graph drawing 
algorithms that can be access via a Java applet that can run locally. Basically, the user 
interface is in Java, and the algorithms are running on the host machine. This doesn’t 
allow for easy inclusion, but it does provide an excellent vehicle to ascertain the abilities 
of graph drawing algorithms. The following is an explanation of the algorithms available 
in GDS. 


1. Giotto. 

Giotto constructs an orthogonal drawing of a graph using a network flow method 
in the orthogonalization phase to obtain the minimum number of bends. Giotto accepts a 
general multigraph as input and augments it to create a cormected graph. The connected 
graph is planarized. Vertices with degree greater than four are expanded into rectangular 
symbols, which are viewed as cycles of degree four to yield a graph with maximum 
degree four. Finally, the graph is passed through orthogonalization and compaction 
phases. The time complexity is 0((N+C)^21og(N+C)) where N is the number of vertices 
in the input graph and C is the number of crossings in the drawing constructed 
[TAMASSIA97]. 

2. Giotto with labels. 

This is a version of Giotto, which draws each vertex as an expanded box large 
enough to fit its label. 
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3. Bend-Stretch. 


Bend-Stretch has the same three steps - planarization, orthogonalization, and 
compaction - as Giotto, and differs only in the method used in the orthogonalization step. 
It adopts the “bend-stretching” heuristic of the Tamassia and Tollis that only guarantees a 
constant number of bends on each edge, but runs in linear time. The time complexity is 
0((N+C)'^21og(N+C)) where N is the number of vertices in the input graph and C is the 
number of crossings in the drawing constructed [TAMASSIA97]. 

4. Pairs. 

Pairs accepts a general multigraph as input, which is augmented to produce a 
coimected graph and then further augmented to produce a biconnected graph. This 
biconnected graph is then drawn according to its orthogonal drawing algorithm. The 
edges added by the augmentation steps are not displayed in the final drawing. The time 
complexity is 0((N+M)log(N+M)) where N is the number of vertices in the input graph 
and M is the number of edges [TAMASSIA97]. 

5. Series Parallel Drawing. 

This algorithm recognizes a series parallel digraph and constmcts an upward 
drawing of it using the delta-drawing algorithm. This is an implementation of “The 
Recognition of Series Parallel Digraphs” by Valdes, Tarjan, and Lawler. Any graph is 
accepted as input; an error message will be displayed if it is not a series parallel digraph 
[TAMASSIA97]. 

6. Sugiyama. 

Sugiyama constmcts a hierarchical drawing of a directed graph according to its 
algorithm. If the input graph is not directed, it is first converted to a directed graph. 
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The algorithm uses a three-step process: first, in a layering step, it assigns vertices 
to horizontal layers. Next, in a crossing-minimization step, it permutes the vertices 
within the same layer to reduce edge-crossings. Finally, in a bend-reduction step, it 
readjusts the position of vertices within each layer to reduce edge-bends 
[TAMASSIA97]. 

7. Column. 

Column is similar to Pairs and differs from it only in the method to optimize the 
number of bends, rows, and columns used in the drawing, once an st-numbering has been 
computed. The method used is the one of Biedl and Kant. The time complexity is 
0(N+M) where N is the number of vertices in the input graph and M is the number of 
edges [TAMASSIA97]. 

8. Ortho Upward. 

Ortho Upward is an algorithm that produces a straight-line orthogonal upward 
drawing of a binary tree. 

9. Ortho Non-Upward. 

Ortho Non-Upward produces a straight-line orthogonal (non-upward) drawing of 
a binary tree. 

10. Planarizer. 

Planarizer is the planarization step of Giotto and constructs a planar embedding of 
the input graph by replacing edge crossing with fictitious vertices. It has time complexity 
0((N+C)^21og(N-i-C)) where N is the number of vertices in the input graph and C is the 
number of crossings in the drawing constructed [TAMASSIA97]. 
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E. GRAPHLET 


Graphlet is a toolkit for graph drawing algorithms. Most applications start with 
an abstract graph structure that has no coordinates. Arranging the nodes and edges in a 
nice fashion is a tedious process for humans. Graph drawing algorithms help users to 
draw graphs. Graphlet’s editor toolkit is implemented with C++, LED A, Tcl/Tk and 
Graphscript. Below are some outputs of Graphlet [HIMSOLT97]. 

Graphlet is available on multiple machines and operating systems. It is available 
on a PC for Windows or Linux. It runs on Sun’s running SunOS or Solaris. Graphlet 
also runs on Hewlett Packard (HP)’s running HP-UX. 

With the diversity of platforms, its intuitive user interface, and the source being 
C++ Graphlet was chosen to display the graphs since the graphical editor was not 
available. 


1, Random Layout. 

Random Layout generates random [x,y] positions for the nodes of a graph. It is 
useful when evaluating different graph algorithms. 
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Figure 3.1: Random Layout 


\ 


2. Spring Embedder with Constraints 


Spring Embedder with Constraints is a straight line layout algorithm that tries to 
maximize space, edge crossing, and angular resolution. 
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Figure 3.2: Spring Embedder with Constraints 


3. Iterative Constraint Spring Embedder 


Iterative Constraint Spring Embedder is similar to Spring Embedder with 
Constraints. However, it iterates the method to produce an orthogonal graph. 
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Figure 3.3: Interative Constraint Spring Embedder 


4. Spring Embedder (GEM) 


GEM is another Spring Embedder. It seeks to minimize space and edge crossing. 
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Figure 3.4; Spring Embedder (GEM) 


5. Spring Embedder (Kamada) 


Kamada is another variation of the Spring Embedder approach. 



Figure 3.5: Spring Embedder (Kamada) 
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6. General Graphs (Tunkelang) 



Figure 3.6: General Graphs (Tunkelang) 


7. DAG 


DAG is a very interesting drawing algorithm. It allows for multiple line 
segments for edges. It also tries to orient the graph to flow down. 
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Figures.?: DAG 
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IV. ANALYSIS OF CAPS GRAPHICAL EDITOR 


A. GENERAL 

The Computer Aided Prototyping System (CAPS) is a software-engineering tool 
designed to take the drudgery out of prototyping systems. A software developer can 
design a software system, test the design, and add timing requirements. Most projects 
become an iterative process of re-visiting the design, making the necessary chtinges to fix 
problems, and then testing the new design. 

After a few iterations, the graphical display of the prototyped system can become 
very difficult to understand, since the graphical display requires the user to layout the 
design. Fixing the design layout by hand is tedious. In the last release of CAPS, Version 
1.1, the graphical editor was almost impossible to use when changing the layout to a more 
meaningful form. The refresh rates made moving a node so cumbersome that the 
software engineer often didn’t want to update a designs layout. 

B. CAPS CHARACTERISTICS 

The set of graphs that CAPS uses is a subset of all graphs. The following are 
characteristics that CAPS graphs will have. These characteristics are important in 
selecting layout algorithms. The most important distinction from the set of all graphs to 
the set of all graphs that CAPS can create is that all CAPS graphs are directed graphs. 

The edges in CAPS represent streams of data. They always flow from a source to 
a destination. The edges are drawn with arrows to depict direction of the data flow. 

These edges are splines. Text on the edge represents the variable name in the CAPS 
PSDL program. CAPS displays one edge per stream. If a call from one object to another 
requires ten streams, then ten edges are drawn with CAPS graphical editor. 

Nodes are represented by two classes of objects with each having a subclass 
resulting in four distinct node representations. Operators are represented with an oval and 
terminators are represented with a rectangle. Each of these types can have subgraphs. 
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This is displayed by putting a double oval or rectangle instead of a single line. Basically, 
the object is halloed if a subgraph exists. Nodes can be color coded to further 
differentiate the class. 
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V. FUTURE REQUIREMENTS OF AUTOMATED LAYOUT TECHNIQUES 

FOR DIRECTED GRAPHS 


A. GENERAL 

Research in the area of automated layout techniques for a directed graph is going 
on in the world, today. Since graphs allow users to visualize many different problems, 
the ability to display these graphs in an aesthetically pleasing manner is highly desirable. 
Every year, since 1992, an annual workshop on graph drawing [GD 92, 93, 94, 95, 96, 
97] is held to improve the capabilities of existing and new algorithms. 

B. RESEARCH AREAS 

There exist many research areas in the field of graph drawing. Because the 
problem in general is NP-Complete, practical algorithms can only be approximations of 
the ideal solution. Also, the ideal solution can vary depending on the problem area being 
visualized. 

Much research is being conducted in general areas of graph drawing and in 
application specific areas. Some of the areas that still require future research that will be 
of particular interest for CAPS are detailed below. 


1. Performance Bounds for Planarization. 

Although crossing minimization is a fundamental issue, non-trivial performance 
bounds have not been found for any heuristic. A guaranteed heuristic would be very 
important both for aesthetic graph drawing and VLSI layout [TAMASSIA94]. 
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2. Simple Planarity Testing. 


The known planarity algorithms that achieve linear time complexity are all 
difficult to understand and implement. This is a serious limitation for their use in 
practical systems. A simple and efficient algorithm for testing the planarity of a graph 
and constructing planar representations would be a significant contribution 
[TAMASSIA94]. 

3. General Strategy for Straight-Line Drawings. 

General strategies have been successfully developed for hierarchical drawings and 
orthogonal grid drawings. These techniques take several aesthetics into account. The 
simplicity of straight-line drawing is very appealing, and a general straight-line drawing 
technique would find immediate applications [TAMASSIA94]. 


4. Dynamic Drawing Algorithms. 

Several graph manipulation systems allow the user to interactively modify a graph 
by inserting and deleting vertices and edges. Data structures that allow for fast 
restructuring of the drawing would be very useful. The time needed to re-compute the 
layout must be small to keep the system from becoming cumbersome, since the algorithm 
would be called every time an update occurred [TAMASSIA94]. 

5. Complexity of Bend Minimization. 

Several issues on the computational complexity of minimizing bends in planar 
orthogonal drawings are open. If the embedding is fixed, bend minimization can be done 
in time 0(n'^21ogn). It would be interesting to improve on the sequential complexity and 
to develop a fast parallel algorithm for the fixed-embedding problem [TAMASSIA94]. 
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6. Angular Resolution of Planar Straight-Line Drawings. 


The angular resolution of a planar straight-line drawing is the minimum angle 
formed by two edges incident on the same vertex. It has been shown that a planar graph 
of degree d has a drawing with angular resolution upper bound of 0(1/d) 
[TAMASSIA94]. 
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VI. NEW AUTOMATED LAYOUT TECHNIQUES FOR CAPS DESIGN AND 

IMPLEMENTATION 


A. GENERAL 

The general idea is to add automated layout techniques to the CAPS graphical 
editor. However, this is easier said than done. Automated techniques for laying out a 
graph are really just estimates of a nice layout. A user will still have to make minor 
adjustments to the graph to fix any minor layout problems. Also, there exist many layout 
techniques that work best depending on the how the user views certain aesthetic 
characteristics and the type of graph being displayed. 

B. CURRENT PROBLEMS 

There are problems related to design and implementation of automated layout 
techniques for the CAPS graphical editor. The first problem is that the CAPS system is 
in a state of transition. The PSDL editor is being re-written to use a new PSDL data type. 
There doesn’t exist any graphical editor for the new PSDL data type. Secondly, the graph 
part of a PSDL data type is private. Without either modifying the PSDL data type to 
export its layout or making the graph functions in the PSDL data type public, there would 
be no way to implement new techniques via the PSDL data type. 

The use of twin lines in CAPS also creates aesthetic problems. Twin lines are 
defined as an edge with the same start node and finish node with equal direction. By 
allowing the user to collapse twin lines for aesthetic reeisons, a more easily understood 
graph is presented. Figure 6.1 demonstrates this point nicely. 
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B. Twin lines 
hidden 



[S1, S2, S3. S4] 



Figure 6.1: Hiding twin lines 


C. DESIGN OF TEST SYSTEM 

The design of this implementation is not optimal. However, given the availablity 
of the graphical editor in the CAPS system it represents a valid workaround. 

Basically, the design is to hook two systems together via some middleware. Since 
the CAPS system outputs a PSDL file, the file could be read and the graph extracted. 

This extracted graph would then be fed into a graphical editor that could display the 
graph. The user interface of this graphical editor would allow the user to run different 
algorithms and modify the parameters of these algorithms. Figure 6.2 gives a graphical 
view of this design. 
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Figure 6.2: Design of test system 


D. IMPLEMENTATION OF TEST SYSTEM 


1. PSDL Data Type. 

The PSDL data type is stored in a PSDL file. These files usually end with the 
suffix “.psdl”. Sinee the graph portion of the PSDL data type is private, one must access 
the graph from a program that uses the PSDL data type without modifying the PSDL data 
type. Public operators available to a user of the PSDL data type do not allow access of 
the graph, therefor only by modifying the PSDL data type can new capabilities be added. 
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2. Middleware. 


The middleware is a program that takes a PSDL file and exports a GML file or 
takes a GML file and updates a PSDL file with the GML file. An example would be the 
following commands: 

% middleware -p <psdl file> <gml file> 

% middleware -g <psdl file> <gml file> 


3. GML File. 

The GML File format is a common format to represent a graph used in many 
graph theory systems. By using this file format, multiple systems could be brought to the 
table to aid in the drawing of graphs. Also, new algorithms could more easily be 
integrated into the system. The complete file format for GML is located in document 
[HIMSOLT96]. 

4. Graphlet system. 

The Graphlet system is freeware that allows a GML file to be read into the 
system. After the graph is input, the user can choose various algorithms, make slight 
modifications to the variables in each algorithm. A user can then make minor 
adjustments to the graph. The new graph can then be saved with the new layout 
[HIMSOLT97]. 

E. DESIGN OF REAL SYSTEM 

In the last month of this writing, a new version of the CAPS graphical editor has 
been available. Since the ideal approach is to connect the layout algorithms directly to 
the graphical editor, the design has changed. Figure 6.3 gives a graphical view of the 
design for this implementation. 
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Figure 6.3: Design of REAL system 


F. IMPLEMENTATION OF REAL SYSTEM 


The implementation of the real system design consisted of porting the LEDA 
libraries to SunOS 4.1.3. After this was accomplished, the C++ LEDA libraries could be 
linked with the edit_graph program. Edit_graph is the executable program for the 
graphical editor. 

The actual linking proved to be more complex than anticipated. CAPS actually 
uses different compilers for different pieces of the system. It is mostly a mixture of 
Ada95, C, and C++. LEDA actually needs to be compiled with a newer version of the 
GNU C++ compiler that CAPS does. In order to link the software, you need to use the 
linker in the newer version of the GNU C++ compiler. 

Two parts of the edit graph need to be modified to use the layout algorithms in 
LEDA. The first part deals with adding menu items to the system that allow a operator to 
invoke different algorithms. The second half deals with converting the PSDL data type 
representation of a graph to a LEDA representation and vis versa. 
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VII. CONCLUSION AND FUTURE RESEARCH 


A. CONCLUSION 

The need for automated layout techniques in CAPS is a very real one. These 
techniques allow a user to clean up the mess associated with iterative edits of the graph. 
The real place to put these algorithms is inside the graphical editor of the CAPS system. 
They should be initiated only by user interaction. 

Since the graphical editor of the CAPS system was not available for the new 
PSDL data type, this could not be done. Instead, middleware was developed to allow the 
two systems to work together. 

The graph part of the PSDL data type should use a standard format consistent 
with the graphing community. This would allow for easy synchronization with leading 
researchers in topology. 

B. FUTURE RESEARCH 

Future research needs to be done on the automated layout of graphs that does a 
better job of placing edge labels. This problem has not been addressed in much detail to 
date. Currently, most algorithms do reasonably well laying out the nodes, node labels, 
and edges. Once these algorithms are finished, they could be modified to handle edge 
labels better. CAPS graphs contain edge labels, operator labels, and maximum execution 
time (MET) labels. Optimizing the layout with all of these objects would increase the 
complexity of the algorithm. 

The use of twin lines in the CAPS system can become very cluttered in the 
graphical display. Allowing the user to toggle the twins to visible or hidden would 
greatly improve the aesthetics of the graph. 

The use of splines for edges needs to be studied. Straight line drawing and 
polyline drawing of edges is the norm. Splines are similar to polyline drawings with the 
comers rounded. If splines do not aid in the comprehension of the graph, then they 
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should be replaced with polylines. Drawing a polyline is much faster than drawing a 
spline. Regardless, the user should be able to choose between splines, straight lines, and 
polylines to represent edges. 


C. CAPS IMPROVEMENT 


1. Spline Drawing 


Splines are currently drawn one point at a time. The actual algorithm is very 
inefficient. It doesn’t take into account the resolution of the system’s display. Duplicate 
points are very possible. This means that the same point will be drawn more than once. 
By computing all the points first, then removing duplicate point, a faster algorithm will 
result. 

Streams are displayed as bold splines. For each of the points in the spline, eight 
additional points are drawn. Basically, all of the original point’s neighbors are also 
drawn point by point. By drawing the bold spline with a filled square of 3x3 pixels 
instead of 9 separate, one pixel draws, the algorithm would be more efficient. 

2. File format 


The current format for the PSDL graph is unique to CAPS. If more conventional 
graph representations were incorporated, then PSDL graphs could be exported to other 
graph systems. This would also allow for improvements to CAPS graphical editor with 
minimal changes by using a standard commonly recognized in the graphing community. 
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APPENDIX A: LEDA LICENSE INFORMATION 


A. LEDA LICENSE 


You are installing the RESEARCH version (LEDA-R) of LEDA that can be used free of 
charge for academic research and teaching. 

FOR ANY COMMERCIAL USE OF THIS SOFTWARE A LICENSE IS 
REQUIRED. ANY KIND OF USE BY A COMPANY OR OTHER NON- 
ACADEMIC INSTITUTION IS CONSIDERED TO BE COMMERCIAL 
USE. YOU ARE BREAKING A LAW WHEN USING LEDA COMMERCIALLY 
WITHOUT OWNING A LICENSE. 

These terms are valid for all LEDA versions following version 3.0. For more information 
about the license terms please contact: 

LEDA Software GmbH 
Postfach 151101 
66041 Saarbruecken 
Germany 

email:leda@mpi-sb.mpg.de 
fax: +49 681 842502 

You are allowed to continue with the installation of LEDA only if you are owner of a 
valid license or if you intend to use LEDA for academic research or teaching. Otherwise, 
you must stop the installation now. 


B. LEDA INFORMATION 


1. LEDA-R-3.5.1 


The new LEDA version available on our ftp server is "LEDA-R-3.5.1". The "R" 
stands for research and has been added to make it distinguishable from the commercial 
version distributed by LEDA Software GmbH. Please read the Changes files for 
information about new features and other changes. 
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LEDA-R-3.5.1 cam be used free of charge in academic research and teaching. 
Licenses for the commercial version "LEDA 3.5.1" are distributed by the LEDA Software 
GmbH. 


2. Differences between LEDA and LEDA-R 


The only difference between LEDA-R and LEDA is that the research version may 
contain additional data types, algorithms or other features which are; 

- experimental (not tested enough or not working on all platforms) 

- of interest only for particular research 

- temporary (may be removed or changed in future versions) 

The commercial version will never contain data types, algorithms or features of 
this kind. It is supposed to stay backward consistent in the sense that old programs should 
work with new versions of LEDA. 


3. Who needs a license? 


- Every organization that uses LEDA and is not an academic research institute or 
school. 

- Everyone who sells programs that are developed using LEDA. 

- Everyone who delivers programs that are developed using LEDA to 
non-academic organizations. 

Holders of a licence for the commercial version are free to also 
use the research version for commercial use. 

Please write to leda@mpi-sb.mpg.de if you want to 

* send bug reports or suggestions 

* get information on the commerical license 

* be on our mailing list 


Subscribe to the LEDA newsgroup comp.lang.c-i-+.leda 
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C. LEDA SOURCE CODE 


•k'k'k'ir'k’k'k'k 

+ 

+ LEDA 3.5.1 
+ 

+ _g_array.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 

+ {fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

ie-k-kie-k-k-k-k-k-kieirieicie-k-k’k-ic’k-ic'k'k-k'k-k'k'k'k'k-^c'k'k'k'k'k'k-k'k'k-k’tc'k’kir-kie-k’k-^ic-tcie'k-k'k’kicieieieie'kic’k-k-k'k'k'k'k'k 

#include <LEDA/graph.h> 


// 


// graph maps and arrays 

// 

// graph_map: base of node/edge/face_map/array 

// 

// by S. Naeher (1995,1996) 

// - 


graph_map: : --graph_map () 

{ if (g && g_index != 0) g->unregister_map(this) 
if (table) delete[] table; 

} 


int graph_map::next_power(int s) const 
{ if (s==0) return 0; 
int p = 1; 

while (p < s) p <<= 1; 
return p; 


void graph_map::re_init_entry(node v) 
{ if {g_index > -1) 

init_entry(v->data[g_index]); 
else 

{ int i = index(v); 
if (i < table_size) 

{ clear_entry(table[i]); 
init_entry(table[i]); 

} 

} 

} 


37 






void graph_map::re_init_entry(edge e) 
{ if (g_index > -1) 

init_entry(e->data[g_index]); 
else 

{ int i = index(e); 
if (i < table_size) 

{ clear__entry (table [i] ) ; 
init_entry(table[i] ) / 

} 

} 

} 


void graph_map::re_init_entry(face f) 
{ if (g_index > -1) 

init_entry(f->data[g_index]); 
else 

{ int i = index(f); 
if (i < table_size) 

{ clear_entry(table[i] ) ; 
init_entry(table[i] ) / 

} 

} 

} 


void graph_map::init_table(GenPtr* start, GenPtr* stop) 

{ if (g_index == -1) 

for(GenPtr* q=start; q < stop; q++) init_entry(*q); 
else 

if (g && g_index > 0) 

{ 

switch (kind) { 
case 0 : { node v; 

forall_nodes(v,*g) init_entry(v->data[g_index]) 
break; 

} 

case 1 : { edge e; 

forall_edges(e,*g) init_entry(e“>data[g_index]) 
break; 

} 

case 2 : { face f; 

forall_faces (f, *g) init_entry (f->data [g__index] ) 
break; 

} 

} 

} 

} 

void graph_map::clear_table() 

{ if (g_index == -1) 

{ GenPtr* stop = table + table^size; 

for(GenPtr* q=table; q < stop; q++) clear_entry(*q); 

} 

else 

if (g && g__index > 0) 

{ switch (kind) { 
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case 


0 : { node v; 

forall_nodes(v,*g) clear_entry(v->data[g_index]) 
break; 

} 

case 1 : { edge e; 

forall_edges(e, *g) clear_entry(e->data[g_index]) 
break; 

} 

case 2 : { face f; 

forall_faces(f,*g) clear_entry(f->data[g_index]) 
break; 

} 

} 

} 


void graph_map::resize_table(int sz) 

{ 

GenPtr* old_table = table; 

GenPtr* old_stop = table + table_size; 

table_size = sz; 
table = new GenPtr[sz]; 

if (table == 0) error_handler (1, ” graph__map: out of memory"); 

GenPtr* p = old_table; 

GenPtr* q = table; 

while (p < old_stop) *q++ = *p++; 

init_table(q,table+sz); 

if (old_table != old_stop) delete[] old^table; 


void graph_map::init(const graph* G, int sz, int k) 

{ 

if (g != G) 

{ if (g && g_index != 0) g->unregister__map (this) ; 
kind = k; 
g = (graph*)G; 

if (g) g_index = g->register__map (this) ; 


if (g_index > -1) 

{ table = 0; 

table_size = 0; 
return; 

} 

clear_table(); 

if (table_size > 0) delete[] table; 
table = 0; 

table_size = next_power(sz); 
if (table_size > 0) 

{ table = new GenPtr[table_size]; 

if (table == 0) error_handler (1, graph_map: out of memory"); 

} 
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} 


graph_map::graph_map(const graph* G, int k) 
{ kind = k; 

g = (graph*)G; 
g_index = 0; 
table = 0; 
table size = 0; 


graph_map::graph_map(const graph* G, int sz, int k) 

{ 

kind = k; 
g = (graph*)G; 
g_index = -1; 

if (g) g_index = g->register_map(this); 

if (g_index > -1) 

{ table = 0; 

table_size = 0; 
return; 

} 

def_entry = 0; 
table = 0; 

table_size = next_power(sz) ; 
if (table_size > 0) 

{ table = new GenPtr[table_size] ; 

if (table == 0) error_handler(1," graph_map: out of memory"); 

) 


graph_map:: graph_map (const graph_map£e M) 

{ kind = M.kind; 
g = M.g; 

if (M.g_index == 0) 

{ g_index = Da¬ 
table = 0; 
return; 

} 

g_index = -1; 

if (g) g_index = g“>register_map(this); 
def_entry = 0; 
table = 0; 

table_size = M.table_size; 
if (table_size > 0) 

{ table = new GenPtr[table_size] ; 

if (table == 0) error_handler(1," graph_map: out of memory"); 
GenPtr* p = table; 

GenPtr* stop = M.table+M.table_size; 
for(GenPtr* q=M.table; q < stop; q++) 

{ *p = *q; 

M.copy_entry(*p); 

P++; 

} 

} 

} 
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graph_map& graph_map::operator=(const graph_map& M) 

{ if (&M == this) return *this; 
clear_table(); 

if (table_size > 0) delete[] table; 

if (g && g_index != 0) g->unregister_map(this) ; 

table = 0; 

kind = M.kind; 

g = M.g; 

if (M.g_index == 0) 

{ g_index = 0; 
table = 0; 
return *this; 

} 

g^index = - 1 ; 

if (g) g_index = g->register_map(this); 
table_size = M.table_size; 
if (table_size > 0) 

{ table = new GenPtr[table_size] ; 

if (table == 0) error_handler (1, ” graph_inap: out of memory”) 
GenPtr* p = table; 

GenPtr* stop = M.table+M.table_size; 
for(GenPtr* q=M.table; q < stop; q++) 

{ *p = *q; 

copy_entry(*p); 

p++; 

} 

} 

return *this; 
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-A--*:****** 

+ 

+ LEDA 3.5.1 
+ 

+ _g_generate.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, ERG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

■k -k -k -k ’k -k -k! 

#include <LEDA/graph.h> 

#include <LEDA/ugraph.h> 

#include <LEDA/vector.h> 

#include <LEDA/matrix.h> 

#include <LEDA/array2.h> 

#include <ctype.h> 

#include <math.h> 

// - 


// some graph generators 

// 

// S. Naeher (1995-1996) 
//- 


void complete_graph(graph& G, int n, bool directed) 

{ 

G.clear(); 

node* V = new node[n]; 

for(int i=0;i<n;i++) V[i] = G.new_node(); 
if (directed) 

{ //memory_allocate_block(sizeof(node_struct),n); 
for(int i=0;i<n;i++) 

for (int j=0; j<n; j ++) G. new__edge (V [i] , V [ j ] ) ; 

} 

else 

{ //memory_allocate_block(sizeof(edge_struct),n*n/2); 
for(int i=0; i<n; i++) 

for(int j=i+l; j<n; j++) G.new_edge(V[i],V[j]); 

} 

delete[] V; 

} 


void grid_graph(graphs G, int n) 
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{ node_array<double> xcoord; 
node_array<double> ycoord; 
grid_graph {G, xcoord,ycoord, n); 

} 

void grid^graph(graphs G, node_array<double>& xcoord, 

node_array<double>& ycoord, int n) 

{ 

array2<node> A(n,n); 

node v; 

int N = n*n; 

int x; 

int y; 

double d = 1.0/(n+1); 

G.clear(); 

xcoord.init(G,N,0); 
ycoord.init(G,N,0); 

for(y=0; y<n/ y++) 
for(x=0; x<n; x++) 

{ A(x,y) = V = G.new_node(); 
xcoord[v] = (x+l)*d; 
ycoord[v] = (y+l)*d; 

) 

for(x=0; x<n; x++) 
for(y=0; y<n; y++) 

{ if (x < n-1) G.new_edge(A(x,y),A(x+1, y) ); 
if (y < n-1) G.new_edge(A(x,y),A(x, y+1)); 

} 


void complete_bigraph(graphs G, int nl, int n2, list<node>S A, 
list<node>S B) 

{ 

G.clear(); 

while (nl—) A. append (G. new__node ()) ; 
while (n2—) B.append(G.new_node()); 

list_item a,b; 
forall_items(a,A) 
forall_items(b,B) 

G.new_edge(A[a],B[b]); 


void user_graph(graphs G) 

{ int n = read__int (" |V| = "); 
int i,j; 

node* V = new node[n]; 

for(j=0; j<n; j++) V[j] = G.new_node()/ 

for(j=0; j<n; j++) 

{ list<int> il; 
int ok = false; 
while (!ok) 
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{ ok = true; 

cout << "edges from [" << j << "] to: "/ 
il.read(); 
forall(i,il) 

if (i<0 II i>=n) 

{ ok=false; 

cout << "illegal node " << i << "\n"; 

} 

} 

forall (i,il) G.new_edge(V[j],V[i]) ; 

} 

G.print()/ 

if (Yes("save graph ? ")) G.write(read_string("file: ")); 
delete [] V; 


void test_graph(graphs G) 

{ 

G.clear(); 
char c; 

do c = read_char("graph: f(ile) r(andom) c(omplete) p(lanar) u(ser) 
") / 

while (c!=*f* && cl^'r* && c!=’c* && c!=*p*&& c!=*u*); 
switch (c) { 

case *f’ : { G.read(read_string("file: ")); 

break; 

} 

case *u* : { user_graph(G); 

break; 

} 

case *c* : { complete_graph(G,read_int("|V| = ")); 

break; 

} 


case ’r’ : { int n = read_int("|V| = "); 

int m = read_int("IEI = "); 
random_graph(G,n,m) ; 
break; 

} 

case *p' : { random_planar_graph(G, read_int("|V| = ")); 

break; 

} 

}//switch 


} 

void test_ugraph(ugraphS G) 

{ 

G.clear(); 
char c; 
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do c = read_char (’’graph: f(ile) r(andom) c(omplete) p(lanar) u(ser): 

") ; 

while (c!=’f' && c!=’r’ && c!=’c’ && c!=’p’&& cl=’u*); 

int i; 
node v; 

switch (c) { 

case *f* : { G. read (read_string (’’file : ")); 

break; 

} 

case *u’ : { int n = read_int(”|V| = ”); 

int j =0; 

node* V = new node[n]; 

for(i=0; i<n; i++) V[i] = G.new_node(); 
forall_nodes(v,G) 

{ list<int> il; 

cout « ’’edges from ” << j++ « " to: 
il.read(); 
forall(i,il) 

if (i>=0&&i<n) G.new_edge(v,V[i]); 
else cerr « ’’illegal node ” << i « ” 

(ignored)\n”; 

} 

G.print(); 

if (Yes (’’save graph ? ”)) G. write (read_string ( "file : 

") ) ; 

delete[] V; 
break; 

} 

case ’c’ : { int n = read_int (’’| V | = "); 

complete^graph(G,n); 
break; 

} 

case 'r' : { int n = read_int (’’| V | = "); 

int m = read_int (’’ I E | = ’’) ; 
random_graph(G,n,m); 
break; 

} 


}//switch 


} 


void test_bigraph(graphs G, list<node>& A, list<node>& B) 

{ 

int a,b; 
int nl = 0; 
int n2 = 0; 
char c; 

do c = read_char("bipartite graph: f(ile) r(andom) c(omplete) u(ser) 
") ; 
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while (c!=’f’ && c!=*r* && c!='c* && c!=»u’)/ 


A. clear(); 

B. clear(); 
G.clear ()/ 


if (c! = *fM 

{ nl = read_int(”|A| = ”)/ 
n2 = read_int(”IBI - ”) ; 

} 


switch (c) { 

case : { G.read(read_string("file : ")); 

node v; 

forall__nodes (v, G) 

if (G.outdeg(v) > 0) A.append(v); 
else B.append(v); 

break; 

} 

case ’u’ : { node* AV = new node[nl+l]; 

node* BV = new node[n2+l]; 

for(a=l; a<=nl; a++) A. append (AV [a] = G. new__node () ) 
for(b=l; b<=n2/ b++) B.append(BV[b] = G.new_node()) 

for(a=l; a<=nl; a++) 

{ list<int> il; 

cout << "edges from " << a << " to: 
il.read(); 
forall(b,il) 

if (b<=n2) G.new^edge(AV[a],BV[b]); 
else break; 
if (b>n2) break; 

} 

delete[] AV; 
delete[] BV; 
break; 

} 

case ’c’ : complete_bigraph(G,nl,n2,A,B); 

break; 

case *r* : { int m = read__int (" | E | = "); 

random_bigraph(G,nl,n2,m,A,B); 
break; 

} 

} // switch 


} 


void cmdline_graph(graphs G, int argc, char** argv) 

{ 

// construct graph from cmdline arguments 
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// no arguments 


if (argc == 1) 

{ test_graph(G); 
return; 

} 

else 

if (argc == 2) // one argument 

{ if (isdigit(argv[l][0])) 

{ cout << "complete graph 1V| = ” << argv[l]; 
newline; 
newline; 

complete__graph (G, atoi (argv [ 1] ) ) ; 

} 

else 

{ cout << "reading graph from file " << argv[l]; 
newline; 
newline; 

G.read(argv[1]) ; 

} 

return; 

} 

else 

if (argc == 3 && isdigit(argv[1][0]) && isdigit(argv[1][0])) 

{ cout << "random graph 1V| = " << argv[l] << " iE| = " << 

argv[2]; 

newline; 

newline; 

random_graph(G, atoi(argv[1]),atoi(argv[2])); 
return; 


error_handler(1,"cmdline_graph: illegal arguments"); 

} 


// 


// triangulated planar graph 
//- 


struct triang_point { 

double x; 
double y; 
node v; 

LEDA__MEMORY (triang_point) 

triang_point(double a=0, double b = 0) { x = a; y = b; v = nil; } 
triang_point(const triang_point& p) { x = p.x; y = p.y; v = p.v; } 

~triang__point () { } ; 

friend bool right_turn(const triang_point& a, const triang_point& b, 
const triang_point& c) 

{ return (a.y~b.y)*(a.x-c.x)+(b.x-a.x)*(a.y-c.y) > 0; } 

friend bool left_turn(const triang_point& a, const triang_point& b, 
const triang_point& c) 
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{ return (a.y-b.y)*(a.x-c.x)+(b.x-a.x)*(a.y-c.y) < 0; } 


friend bool operator==(const triang_point& a, const triang_point& b) 
{ return a.x == b.x && a.y == b.y; } 

friend ostream& operator<<(ostreamS out, const triang_point& p) 

{ return out << p.x << ” " << p.y; } 

friend istream& operator»(istream& in, triang_point& p) 

{ return in » p.x >> p.y; } 

friend int compare(const triang_point& p, const triang_point& q) 

{ int c = compare(p.x,q.x); 

if (c==0) c = compare(p.y, q.y) ; 
return c; 

} 


}; 


void triangulated_planar_graph(graphs G, list<node>& outer_face, 

node_array<double>& xcoord, 
node_array<double>& ycoord, int 
n) 


G.clear()/ 

list<triang__point> L; 
while(n—) 

{ double X = rand_int(0,1000000)/lOOOOOO.0; 
double y = rand_int(0,1000000)/lOOOOOO.0; 

L.append(triang_point(x, y)); 

} 

L.sort(); // sort triang_points lexicographically 

list<triang_point> CH; 
list_item last/ 
triang_point p,q; 


// eliminate multiple triang_points 

list_item it; 
forall_items(it, L) 

{ list__item itl = L.succ(it); 

while (itl != nil && L[itl] == L[it]) 
{ L.del(itl); 

itl = L.succ(it); 

} 

} 


n = L.length(); 

xcoord.init(G,n,0); 
ycoord.init(G,n,0); 

forall_iterns(it,L) 
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{ node V = G.new_node() ; 
xcoord[v] = L[it].x; 
ycoord[v] = L[it].y; 

L [it] . V = v; 

} 

// initialize convex hull with first two points 

p = L.pop(); 

CH.append(p); 

while (L.headO == p) L.popO; 

q = L.pop(); 

last = CH.append(q); 

G.new_edge(p.v,q.v); 


// scan remaining points 

forall(p,L) 

{ 


node V = p.v; 

G.new_edge(v,CH[last].v); 

// compute upper tangent (p,up) 

list_item up = last; 

list_item it = CH.cyclic_succ(up); 

while (left_turn(CH[it],CH[up],p)) 
{ up = it; 

it = CH.cyclic_succ(up); 

G.new_edge(v,CH[up].v); 

} 


// compute lower tangent (p,low) 

list_item low = last; 
it = CH.cyclic_pred(low); 

while (right_turn(CH[it] , CH[low],p)) 
{ low = it; 

it = CH.cyclic_pred(low); 

G.new_edge(v,CH[low].v); 

} 


// remove all points between up and low 

if (up 1= low) 

{ it = CH.cyclic_succ(low); 

while (it != up) 

{ CH.del(it); 

it = CH.cyclic_succ(low); 


49 


} 

} 

// insert new point 
last = CH.insert(p,low); 


} 

outer_face.clear(); 

forall(p,CH) outer_face.append(p.v); 


} 


void triangulated_planar_graph(graphs G, int m) 

{ node_array<double> xcoord; 
node_array<double> ycoord; 
list<node> L; 

triangulated_planar_graph(G,L,xcoord,ycoord,m); 

} 


static bool tutte_embed(const graphs G, const node_array<bool>S fixed, 

node_array<double>S xpos, node_array<double>S 

ypos) 

{ node v,w; 
edge e; 


list<node> other_nodes; 
forall_nodes(v, G) 

if(!fixed[v]) other_nodes.append(v); 
node_array<int> ind(G); // position of v in other_nodes and A 

int i = 0; 

forall(v,other nodes) ind[v] = i++; 


int n = 

= other_nodes.size() ; 

// 

vector 

coord(n); 

// 

vector 

rhs(n); 

// 

matrix 

A(n, n) ; 

// 


#other nodes 

coordinates (first x then y) 

right hand side 

equations 


// initialize non-zero entries in matrix A 
forall(v,other_nodes) 

{ 

double one_over_d = 1.0/double(G.degree(v))/ 
forall_inout_edges(e,v) 

{ 

// get second node of e 

w = (v == source(e)) ? target(e) : source(e); 

if(!fixed[w]) A(ind[v],ind[w]) = one_over_d; 

} 

A(ind[v],ind[v]) = -1; 


if(!A.det()) return false; 

// compute right hand side for x coordinates 
forall(v,other_nodes) 

{ rhs[ind[v]] = 0; 
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double one_over_d = 1.0/double(G.degree(v)); 
forall_inout_edges(e,v) 

{ // get second node of e 

w = (v == source(e)) ? target(e) : source(e); 

if(fixed[w]) rhs[ind[v]] (one_over_d*xpos[w]); 

} 

} 

// compute X coordinates 
coord = A.solve(rhs); 

forall(V,other_nodes) xpos[v] - coord[ind[v]]; 

// compute right hand side for y coordinates 
forall(V,other_nodes) 

{ rhs[ind[v]] =0; 

double one_over_d = 1.0/double(G.degree(v)); 
forall_inout_edges(e, v) 

{ // get second node of e 

(v == source (e)) ? target (e) : source (e); 

if(fixed[w]) rhs[ind[v]] (one_over_d*ypos[w])/ 

} 


// compute y coordinates 
coord = A.solve(rhs); 

forall(v,other_nodes) ypos[v] = coord[ind[v]]; 
return true; 


void triangulated_planar_graph(graphs G, node_array<double>& xcoord, 

node_array<double>& ycoord,^int 
h) 

{ list<node> L; 

triangulated_planar_graph(G,L,xcoord,ycoord, n); 

if (n > 128) return; 

node_array<bool> fixed(G,false); 

double step = 6.2832/L.length(); 
double alpha = 0; 
node v; 
forall(V,L) 

{ xcoord[v] = cos(alpha); 
ycoord[v] = sin(alpha); 
alpha+=step; 
fixed[v] = true; 

} 

tutte_embed(G,fixed,xcoord,ycoord); 

} 
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+ 

+ LEDA 3.5.1 
+ 

+ _g_gitilio.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 

+ (fax +49 681 31104) . 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

ir^ic'k’k'k'k^'k-k'k-k^'k'k'k-k-k-k-k-k-k'k'k-k'k-k'k'k-k'k-k-k'k'k'k-k'k-k-k-k-k'k-k-k'k-ir^'k-k'k-k'k-k-k^'k-k'k'k'k-k-k-k'k'k'k'k^'k'k^ 


*★*★★**! 

II - // 

// writing LEDA graphs in GML format // 

// reading LEDA graphs in GML format // 

// // 

// David Alberts (1996) // 

// David Alberts (1997) new version, accepts unknown lists, etc. // 

// no more lex/yacc // 

// - // 


#include<LEDA/graph.h> 
#include<LEDA/stream.h> 
#include<LEDA/gml_graph.h> 


bool graph::write_gml(string outfile, 

void (*node cb)(ostream&,const 


node), 


graph*. 


void {*edge_cb)(ostream&,const graph*, 

edge)) const 

// writes a graph description in GML format to outfile. 

// If an error occurs, false is returned. 

{ 


file ostream out(outfile); 


const 

const 


if(out.fail 0) return false; 

else return write_gml(out,node_cb,edge_cb); 


bool graph::write 
node), 


gml(ostream& out, 

void (*node cb)(ostream&,const 


graph*, 


void (*edge_cb)(ostream&,const graph*, 

edge)) const 

// writes a graph description in GML format to outfile. 

// If an error occurs, false is returned. 

{ 


if(out.fail 0) return false; 


const 

const 


string void_str("void”); 


out 

<< 

"Creator " « 

<< 

"LEDA write gml" 

« » " * 

« " 

out 

<< 

"graph [\n\n"; 





out 

« 

" directed " « 

(is 

directed 0 ? 1 : 

0) « 

"\n"; 
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out « "\n”; 


node v; 

if((string(node_type0) != void_str) || node_cb) 

{ 

forall_nodes(v,*this) 

{ 

out << " node [\n”; 

out << ” id ” << index(v) << "\n”; 
if(string(node_type0) != void_str) 

{ 

out << " parameter " << *”* << get_node_entry_string(v); 
out << << ”\n"; 

} 

if(node_cb) ('^node_cb) (out, this, v) ; 
out << " ]\n"/ 

} 

} 

else 

forall_nodes(V,*this) out « " node [ id ” << index(v) << ” ]\n" 
out << "\n"/ 
edge e; 

if((string(edge_type0) != void_str) M edge_cb) 

{ 

forall_edges(e,*this) 

{ 

out « " edge [\n"; 

out << " source " << index(source(e)) << ”\n"; 

out << " target " << index(target(e)) « ”\n"; 

if(string(edge_type0) != void_str) 

{ 

out << " parameter " << << get_edge_entry_string(e); 

out « « "\n"; 

} 

if(edge_cb) (■*^edge_cb) (out, this, e) ; 
out << ” ]\n"; 

} 

} 

else 

{ 

forall_edges(e,*this) 

{ 

out << " edge [ source ” << index(source(e)) « " 
out << "target " << index(target(e)) << " ]\n”; 

} 

} 

out « "\n]\n"/ 
return true; 


bool graph::read_gml(string s) 

{ 

gml_graph* parser = new gml_graph(*this,s.cstring() ) ; 
bool ok = !parser->errors(); 
delete parser; 
return ok; 
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bool graph:: read_grtil (istream& in) 

{ 

ginl_graph* parser = new gml_graph (*this, in) 
bool ok = !parser->errors() ; 
delete parser; 
return ok; 
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•*-■*■■*■•*■ + ■*■■*■■*- 
+ 

+ LEDA 3.5.1 
+ 

+ _g_inout.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104) . 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

+ ! 


// 


// graph i/o 


// 

// S. Naeher 
//- 


(last modified: December 1996) 


tinclude <LEDA/graph.h> 
tinclude <LEDA/stream.h> 
tinclude <ctype.h> 

#include <string.h> 

const char delim = * I'; 

void graph::write(string file_name) const 
{ ofstream out(file_name); 
if (out.fail 0) 

error_handler(1,string("graph::write() cannot open file 
%s",file_name)); 
write(out); 

} 


void graph:-.write (ostream& out) const 

{ 

int* A = new int[max_n_index+l] ; 
int* B = new int[max_e_index+l]; 

for (int i=0; i<= max__e_index; i++) B[i] = 0; 

// nodes get numbers from 1 to |V| 


int 

n 

count = 1; 



int 

e 

count = 1; 



out 

« 

"LEDA.GFLAPH” 

« 

endl; 

out 

« 

node_type () 

« 

endl / 

out 

« 

edge type() 

« 

endl; 
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out « v_list.length() << endl; 
node v; 

forall_nodes(v,*this) 

{ out << delim « *{'; 

write_node_entry(out , v->data[0]) ; 
out << *}* << delim « endl; 

A[index{v)] = n_count++; 

} 

out « nuinber_of_edges () << endl; 

forall_nodes(v,*this) 

{ edge e; 

int s = A[index(v)]; 
forall_adj_edges(e^ v) 

{ if (source(e) != v) continue; // necessary for ugraphs 

int t = A[index(target(e))]; 
int r = (e“>rev) ? B[index(e“>rev)] : 0; 

out « s « ” " « t « " " « r « " « delim << * { * ; 

write_edge_entry(out,e->data[0]); 
out << *}* << delim << endl; 

B[index(e)] = e_count++; 

} 

} 

delete[] A; 
delete[] B; 


int graph::read(string file_name) 
{ ifstream in(file_name); 
if (in.fail()) return 1; 
return read(in); 


static void read_data_entry(istream& in, char* buf, int buf_sz) 
{ char* p = buf-1; 
char c; 

do in.get(c); while (isspace(c) ) ; 
if (c != delim) 

{ in.putback(c); 

string line = read_line(in) ; 
strcpy(buf,line.cstring()) ; 
return; 

} 

in.get(c); 
if (c != »{*) 

error_handler(1,"graph::read(): error in graph format."); 

int nested = 1; 

while (nested) 

{ in.get(c); 

if (c == delim && p >= buf) 

{ if (*p == »}*) 
nested--; 
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else 

{ if (buf_sz— <= 0) 

error_handler(1,"graph::read: data overflow"); 
*++p = c; 
in.get(c); 

if (c == ’{’) nested++; 

} 

} 

if (nested) 

{ if (buf_sz— <= 0) 

error_handler(1,"graph::read: data overflow"); 
*++p = c; 


} 

*p = »\0'; 

} 


int graph::read(istream& in) 

{ 


clear (); 

int result = 0; 
int n,i,v,w,r; 

string this_n_type = node_type(); 
string this_e_type = edge_type(); 

string d_type,n_type,e_type; 

in » d_type; 
in >> n_type; 
in >> e_type; 
in >> n; 

if (d_type != "LEDA.GRAPH") return 3; 

read_line(in); 

node"*^ A = new node[n-(-l]; 

char data_str[1024 ] ; 

if (this_n_type == "void" | | n_type != this_n_type) // do not read 
node info 

{ for (i=l; i<=n; i++) 

{ A[i] = new_node(); 

read_data_entry(in,data_str,1024); 

} 

if (n_type != this_n_type) result =2; // incompatible node 

types 

} 

else 

if (this_n_type == "string") 
for (i=l; i<=n; i++) 

{ A[i] = new_node(0); 

read_data_entry(in,data^str, 1024) ; 

A[i]->data[0] = leda_copy(string(data_str)); 
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} 

else 

for (i=l; i<=n; i++) 

{ A[i] = new_node(0); 

read_data_entry(in,data_str,1024); 
istrstream str_in(data_str,strlen(data_str)); 
read_node_entry(str_in,A[i]->data[0]); 

} 


in » n; // number of edges 

edge* B = new edge[n+l]; 

if (this_e_type == "void" || e_type != this_e_type) // do not read 
edge info 

{ if (e_type != this_e_type) result = 2; // incompatible edge 

types 

for (i=l; i<=n; i++) 

{ in » V >> w » r; 

edge e = new_edge(A[v],A[w]); 
read_data_entry(in,data_str,1024); 

B [ i ] = e ; 

if (r > 0) set reversal(e,B[r]); 


else 

if (this_e_type == "string") 
for (i=l; i<=n; i++) 

{ in » V » w >> r; 

edge e = new_edge(A[v],A[w],GenPtr(0))/ 
read_data_entry(in,data_str,1024); 
e->data[0] = leda_copy(string(data_str)); 

B[i] = e; 

if (r > 0) set_reversal(e,B[r]); 

} 

else 

for (i=l; i<=n; i++) 

{ in >> V >> w » r; 

edge e = new_edge(A[v],A[w],GenPtr(0)); 
read_data_entry(in,data_str,1024); 
istrstream str_in(data_str,strlen(data_str)); 
read_edge_entry (str_in, e-‘>data [0] ) ; 

B [ i ] = e ; 

if (r > 0) set_reversal(e, B[r]); 

} 

delete[] A; 
delete[] B; 

return result; 


void graph::print_node(node v,ostream& o) const 
{ if (superO != 0) 

super()->print_node(node(graph::inf(v)),o); 
else 

{ o « "[" « index (V) «"]" ; 

print_node_entry(o, v->data[0]); 

} 
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void graph::print_edge(edge e,ostream& o) const 
{ if (super() != 0) 

super()->print_edge(edge(graph::inf(e)) , o) ; 
else 

{ o << "[" << index(source(e)) << 

o << ((undirected) ? "==" : 
print_edge_entry(o,e->data[0] ) ; 
o << ((undirected) ? "==" : "—>"); 
o « « index(target(e)) « 

} 


void graph::print(string s, ostreamS out) const 
{ node v; 
edge e; 

out << s << endl; 
forall_nodes(v,*this) 

{ print_node(v,out); 
out « " : 

forall_adj__edges (e, v) print_edge (e, out) ; 
out « endl; 

} 

out << endl; 


// convert node and edge entries into a string and vice versa 

string graphget_node_entry_string(node v) const 
{ ostrstream out; 

write_node_entry(out,v->data[0] ) ; 

out << ends; 

char* p = out.strO; 

string s(p); 

delete[] p; 

return s; 

} 

string graph:: get__edge_entry_string (edge e) const 
{ ostrstream out; 

write_edge_entry(out,e->data[0]); 

out << ends; 

char* p = out.strO; 

string s(p); 

delete[] p; 

return s; 

} 


void graph:: set__node_entry (node v, string s) 
{ clear_node_entry(v->data[0] ) ; 

if (strcmp(node_type0,"string") == 0) 
v->data[0] = leda_copy(s); 
else 

{ istrstream in(s.cstring()); 
read_node_entry(in,v->data[0]); 

} 
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void graph::set_edge_entry{edge e, string s) 
{ clear_edge_entry(e->data[0]); 

if (strcmp(edge_type0,"string”) == 0) 
e->data[0] = leda_copy(s); 
else 

{ istrstream in(s.cstring()); 
read_edge_entry(in,e“>data[0] ) ; 

} 

} 
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+ 

+ LEDA 3.5.1 
+ 

+ _g_map.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104) . 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

j 

#include <LEDA/graph.h> 

// reversal edges 

edge graph::face_cycle_succ(edge e) const 
{ return cyclic_adj_pred(reversal(e)) ; } 

edge graph::face_cycle_pred(edge e) const 
{ return reversal(cyclic_adj_succ(e)); } 

edge graph::succ_face_edge(edge e) const 
{ return cyclic_adj_pred(reversal(e)); } 

edge graph::pred_face_edge(edge e) const 
{ return reversal(cyclic_adj_succ(e)) ; } 


void graph::set_reversal(edge e, edge r) 

{ if ( source(e) != target(r) || target(e) != source(r) ) 

error_handler(l,"graph::set_reversal: edges are not reversals of each 
other"); 

e->rev = r; 
r->rev = e; 

} 


static int map_edge_ordl(const edge& e) { return index(source(e)); } 
static int map_edge_ord2(const edge& e) { return index(target (e)) ; } 

bool graph::make_map() 

{ 

// computes for every edge e = (v,w) in G its reversal r = (w,v) 

// in G ( nil if not present). Returns true if every edge has a 
// reversal and false otherwise. 

int n = max_node_index() ; 

int count = 0; 

list<edge> Ell = all_edges(); 
list<edge> E12 = Ell; 

edge e; 
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forall(e,Ell) e->rev = 0; 

Ell.bucket_sort(0,n,&map_edge_ord2); 
Ell.bucket_sort(0,n,&map_edge_ordl); 
E12.bucket_sort(0,n,&map_edge_ordl); 
E12.bucket_sort(0,n,&map_edge_ord2)/ 


// merge Ell and E12 to find corresponding edges 

while (! Ell.emptyO && I E12.empty()) 

{ edge e = Ell.head{)/ 
edge r = El2.head(); 
if (target(r) == source(e)) 
if (source(r) == target(e)) 

{ e->rev = r; 

El2.pop(); 

Ell.pop(); 
count++; 

} 

else 

if (index(source(r)) < index(target(e))) 
E12.pop(); 

else 

Ell.pop(); 


else 

if (index(target(r)) < index(source(e))) 
E12.pop(); 

else 

Ell.pop(); 


} 

return (count == nuniber_of_edges () ) ? true : false; 


void graph::make_map(list<edge>& R) 

{ if (make_map()) return; 

list<edge> el = all_edges(); 
edge e; 
forall(e,el) 

{ if (e->rev == nil) 

{ edge r = new_edge(target(e),source(e) ) ; 
e~>rev = r; 
r->rev = e; 

R.append(r); 

} 

} 


extern bool PLANAR(graphs, bool=false); 

void graph::make_planar_map() 

{ if (1 graph: :make_map0) 
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error_handler(1,"graph::make_planar_map: graph is not 
bidirected"); 

if (!PLANAR(*this,true)) 

error_handler(1,"graph::make_planar_map: Graph is not planar.") 
compute_faces(); 


face graph::new_face(GenPtr i) 
{ copy_face_entry(i) ; 
return add__face (i) ; 

} 

face graph::new_face() 

{ GenPtr i = 0; 

init_face_entry(i); 
return add_face(i); 

} 


list<edge> graph::adj_edges(face f) const 
{ list<edge> result(f->head); 

edge el = face_cycle_succ(f->head) ; 
while (el!=f“>head) 

{ result.append(el) ; 

el = face_cycle_succ(el) ; 

} 

return result; 

} 

list<node> graph::adj_nodes(face f) const 
{ list<node> result(source(f->head)); 
edge el = face_cycle__succ (f->head) ; 
while (el!=f->head) 

{ result.append(source(el)); 
el = face_cycle_succ(el); 

} 

return result; 

} 

list<face> graph::adj_faces(node v) const 
{ list<face> result; 
edge e; 

forall_out__edges (e, v) result. append (adj_face (e) ) ; 
return result; 

} 


void graph::print_face(face f) const 
{ cout << string("F[%2d]", index(f)); 
cout << "("; 

print_face_entry(cout,f->data[0]); 
cout « "): "; 
edge e; 

forall_face_edges(e,f) 
cout << string("[%2d]", index(target(e))) ; 

} 
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void graph::compute_faces() 


{ 

del_all_faces(); 

FaceOf = new graph_map(this, 1,0) ; 
edge e; 

forall_edges(e,*this) 

{ if (e->rev == nil) 

error_handler(1,"graph::computeffaces: no map (reversal edge 
missing)"); 

access_face(e) = nil; 

} 


forall_edges(e,*this) 

{ if (access_face(e) != nil) continue; 

face f = new_face(); 
f->head = e; 
edge el = e; 
int count = 0; 
do { access_face(el) = f; 

el = face_cycle_succ(el); 
count++; 

} while (el != e); 
f->sz = count; 

} 


edge graph::split_map_edge(edge e) 

{ 

/* splits edge e and its reversal by inserting a new node u (node_inf) 
e e rr 

- > - > - > 

(v) (w) ====> (v) (u) (w) 

< - < - < - 

r er r 

returns edge rr 

*/ 

edge r = e->rev; 
if (r == nil) 

error_handler(1,"graph;:split_map_edge(e): reversal of edge e 
missing."); 

node V = source (e); 
node w = target (e); 
node u = new_node(); 

// remove e and r from corresponding in-lists 
W“>del_adj_edge(e,1,1); 
v->del_adj_edge(r,1,1); 


// insert e and r in in-list of u 
e->term[l] = u; 
r->term[l] = u; 
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u->append_adj_edge(e, 1 , 1); 
u~>append_adj_edge(r, 1 , 1) ; 

// create reverse edges rr and re 
edge rr = graphnew_edge(u, w); 
edge er = graph::new_edge(u, v)/ 

set_reversal(e,er); 
set_reversal(r,rr) / 

access_face(rr) = access_face(e) ; 
access_face(er) = access_face(r); 

return rr; 

} 


edge graph::new_map_edge(edge el, edge e2) 
{ edge e = graph::new_edge(el,source(e2)) ; 
edge r = graph::new_edge(e2,source(el)) ; 
set_reversal(e,r); 
return e; 

} 


edge graph::split_face(edge el, edge e2) 

{ 

face fl = access_face(el) ; 
face f2 = access_face(e2) ; 

if (fl != f2) 

error_handler(1,"planar_map::new_edge: new edge must lie in a 
face.”); 

f2 = new_face(); 

edge x = graph::new_edge(el,source(e2)) / 
edge y = graph:;new_edge(e2, source(el)) ; 
set_reversal(x,y)/ 

fl->head - x; 
f2->head = y; 

access_face(x) = fl; 

do { access_face(y) = f2; 

y = face_cycle_succ(y); 

} while (y ! == f2~>head) ; 

return x; 

} 


list<edge> graph::triangulate_map() 

{ 

/* G is a planar map. This procedure triangulates all faces of G 

without introducing multiple edges. The algorithm was suggested by 
Christian Uhrig and Torben Hagerup. 

Description: 
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Triangulating a planar graph G, i.e., adding edges 
to G to obtain a chordal planar graph, in linear time: 

1) Compute a (combinatorial) embedding of G. 

2) Step through the vertices of G. For each vertex u, 
triangulate those faces incident on u that have not 
already been triangulated. For each vertex u, this 
consists of the following: 

a) Mark the neighbours of u. During the processing 
of u, a vertex will be marked exactly if it is a 
neighbour of u. 

b) Process in any order those faces incident on u 
that have not already been triangulated. For each such 
face with boundary vertices u=x_l,...,x_n, 

I) If n=3, do nothing; otherwise 

II) If x_3 is not marked, add an edge {x_l,x_3}, 
mark x_3 and continue triangulating the face 
with boundary vertices x_l,x_3,x_4,...,x_n. 

III) If x_3 is marked, add an edge {x_2,x_4} and 
continue triangulating the face with boundary 
vertices x_l,x_2,x_4,x_5,...,x_n; 

c) Unmark the neighbours of x_l. 

Proof of correctness: 

A) All faces are triangulated. 

This is rather obvious. 

B) There will be no multiple edges. 

During the processing of a vertex u, the marks on 
neighbours of u clearly prevent us from adding a multiple 
edge with endpoint u. After the processing of u, such an 
edge is not added because all faces incident on u have 
been triangulated. This takes care of edges added in 
step II). 

Whenever an edge {x_2,x__4} is added in step III), the 
presence of an edge {x__l,x_3} implies, by a topological 
argument, that x__2 and x_4 are incident on exactly one 
common face, namely the face currently being processed. 
Hence we never add another edge {x_2,x__4}. 


node v; 
edge x; 
list<edge> L; 

node_array<int> marked(*this,0); 
if ( !make_map() ) 

error_handler(1,"TRIANGULATE_PLANAR_MAP: graph is not a map.") 

forall__nodes (v, *this) 

{ 

list<edge> El = adj_edges(v); 
edge e,el,e2,e3; 
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forall(el,El) marked[target(el)]=1/ 

forall(e,El) 

{ 

el = e; 

e2 = face_cycle_succ(el)/ 
e3 = face_cycle_succ(e2)/ 

while (target(e3) != v) 

// el,e2 and e3 are the first three edges in a clockwise 
// traversal of a face incident to v and t(e3) is not equal 
// to V. 

if ( Imarked[target(e2)] ) 

{ // we mark w and add the edge {v,w} inside F, i.e., after 
// dart el at v and after dart e3 at w. 

marked[target(e2)] = 1; 

L.append(x = new_edge(e3,source(el))); 

L.append(el = new_edge(el,source(e3))) ; 
set_reversal(x,el); 
e2 = e3; 

e3 = face_cycle_succ(e2); 

} 

else 

{ // we add the edge {source (e2),target(e3)} inside F, i.e., 
// after dart e2 at source(e2) and before dart 
// reversal_of[e3] at target(e3). 

e3 = face_cycle_succ(e3); 

L.append(x = new_edge(e3,source(e2))) ; 

L.append(e2 = new_edge(e2,source(e3) )) ; 
set_reversal(x,e2) ; 

} 

//end of while 

} //end of stepping through incident faces 
node w; 

forall_adj_nodes(w,v) marked[w] = 0; 

} // end of stepping through nodes 
return L; 


) 


face graph::join_faces(edge x) 

{ 

edge y = reversal(x); 
if (y == nil) 

error^handler(1, ”join_faces: graph must be a map.”); 

if (access_face (x) == nil | I access_face (y) =^= nil) 

error_handler(1,”join_faces: no face associated with edges."); 

edge e = face_cycle_succ(y); 
face FI = adj_face(x); 
face F2 = adj_face(y); 
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if (FI != F2) 

{ edge e = face_cycle__succ (y) ; 
Fl->head = e; 
while ( e 1= y ) 

{ access_face(e) = FI; 

e = face_cycle_succ(e); 

} 

clear_face_entry(F2->data[0]); 
del_face(F2); 

} 

else 

{ e = face_cycle_succ(e); 

if (e != y) // no isolated edge 
Fl->head = e; 
else 

{ clear_face_entry(Fl->data[0]) ; 
del_face(FI); 

FI = F2; 

} 

} 

graph::del_edge(x); 
graph::del_edge(y); 

return FI; 


void graph::make_bidirected(list<edge>& L) 
{ Make_Bidirected(*this,L); } 

bool graph::is_bidirected() const 
{ edge_array<edge> rev(*this,0); 
return Is_Bidirected(*this^rev); 

} 

bool graph::is_map() const 
{ return Is_Map(*this); } 
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•k'k'k'k'k'k'k'k 

+ 

+ LEDA 3.5.1 
+ 

+ _g_misc.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, ERG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

*•*:★**■*■★ ^ 

#include <LEDA/graph.h> 

#include <LEDA/ugraph.h> 

#include <LEDA/graph_alg.h> 

// - 


// S. Naeher 

// last modified ( April 1997) 

// - 


node_array<int>* num_ptr; 

static int source_num(const edge& e) { return (*num_ptr)[source(e)]/ } 
static int target_num(const edge& e) { return (*num_ptr)[target(e)]; } 


bool Is_Simple(const graphs G) 

{ 

// return true iff G is simple, i.e, has no parallel edges 
list<edge> el= G.all_edges(); 
if (el.empty()) return true; 


int n = 0; 

node_array<int> num(G); 
node v; 

forall_nodes(V,G) num[v]= n++; 
num_ptr= Snum; 

el.bucket_sort(0,n-1,&source_num); 
el.bucket_sort(0,n-1,&target_num) ; 

edge eO = el.popO; 
edge e; 
forall(e,el) 

{ if (source(eO) == source(e) && target(eO) == target(e) ) 

return false; 


69 






} 


else 

eO = e; 


return true; 


} 


list<node> Delete_Loops(graphs G) 
{ list<edge> loops; 
list<node> L; 
edge e; 

forall_edges(e,G) 

{ node V = source(e); 
node w = target(e); 
if (v == w) 

{ L.append(v); 
loops.append(e); 

} 

} 

forall(e,loops) G.del_edge(e); 
return L; 

} 


list<edge> Make_Simple(graphs G) 

{ 

list<edge> L; 

//use bucket sort to find and eliminate parallel edges 
list<edge> el = G.all_edges(); 
if (el. empty 0) return L; 


node_array<int> num(G); 
int n = 0; 
node v; 

forall_nodes(V,G) num[v] = n++; 
num_ptr = Snum; 

el.bucket_sort(0,n-1,Ssource_num); 
el.bucket_sort(0,n-1,Starget_num); 

bool deleted = false; 
edge eO = el.popO; 

edge e; 
forall(e,el) 

if (source(eO) == source(e) SS target(eO) == target(e)) 
{ G.del_edge(e); 

if (!deleted) L.append(eO); 
deleted = true; 

} 

else 

{ deleted = false; 
eO = e; 

) 
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} 


return L; 


static int edge_ordl(const edge& e) { return index(source(e)); } 
static int edge_ord2(const edge& e) { return index(target(e)); } 

bool Is_Bidirected(const graphs G, edge_array<edge>& reversal) 

{ 

// computes for every edge e = (v,w) in G its reversal reversal[e] 
(w, v) 

// in G ( nil if not present). Returns true if every edge has a 
// reversal and false otherwise. 

int n = G.max_node_index(); 
int count = 0; 

edge e,r; 

forall_edges(e,G) reversal[e] = 0; 

list<edge> El = G.all_edges() ; 

El.bucket_sort(0,n,&edge_ord2); 

El.bucket_sort(0,n,&edge_ordl); 

list<edge> Ell = G.all_edges() ; 

Ell.bucket_sort(0,n,&edge_ordl); 

Ell.bucket_sort(0,n,&edge_ord2); 


// merge El and Ell to find corresponding edges 

while (! El.emptyO && ! Ell. empty ()) 

{ e = El.head(); 
r = Ell.head 0; 

if (target(r) == source(e)) 
if (source(r) == target(e)) 

{ reversal[e] = r; 

Ell.popO ; 

El.pop(); 
count++; 

} 

else 

if (index(source(r)) < index(target(e))) 
Ell.pop(); 

else 

El.popO ; 


else 

if (index(target(r)) < index(source(e))) 
Ell.pop(); 

else 

El.pop(); 


} 

return (count == G.number_of_edges()) ? true : false; 
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void Mak;e_Bidirected (graphs G, list<edge>& R) 

{ 

// make graph bi-directed by inserting reversal edges 
// appends new edges to R 

edge_array<edge> rev(G,nil); 

if (Is_Bidirected(G,rev)) return; 

// build list L of edges having no reversals 

list<edge> L; 
edge e; 

forall_edges(e, G) 
if (rev[e] == nil) L.append(e); 

// insert missing reversals 
forall (e,L) 

{ edge r = G.new_edge(target(e),source(e)); 

R.append(r); 

} 


list<edge> Make_Bidirected(graphs G) 
{ list<edge> R; 

Make_Bidirected(G,R); 
return R; 

} 


static void dfs(node v, intS count1, ints count2, node_array<int>S 
dfsnum, 

node_array<int>S 

compnum) 

{ dfsnum[v] = ++countl; 
edge e; 

forall_adj_edges(e, v) 

{ node w = target(e); 
if (dfsnum[w] == 0) 

dfs(w,count1,count2,dfsnum,compnum); 

} 

compnum[v] = ++count2; 


bool Is_Acyclic(const graphs G, list<edge>S back) 

{ 

//compute dfs and completeion numbers 

node_array<int> dfsnum(G,0); 

node_array<int> compnum(G,0); 

int count1 = 0; 

int count2 = 0; 

node v; 

forall_nodes(v,G) 
if (dfsnum[v] == 0) 

dfs(v,countl,count2,dfsnum,compnum); 
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// compute back edges 
back.clear(); 
edge e; 

forall_edges(e,G) 

{ node V = source(e); 
node w = target(e); 

if (v == w II (dfsnum[v] > dfsnum[w] && compnum[v] < compnum[w])) 
back.append(e); 

} 

return back.empty(); 


bool Is_Acyclic(const graphs G) 
{ list<edge> dummy; 

return Is_Acyclic(G,dummy); 

} 


void Make_Acyclic(graphs G) 

{ list<edge> back; 

Is_Acyclic(G,back); 
edge e; 

forall(e,back) G.del_edge(e); 


static void dfs(const graphs G, node v, node_array<bool>S reached, intS 
count) 

{ reached[v] = true; 
count++; 
edge e; 

forall_inout_edges(e,v) 

{ node w = G.opposite(v, e); 

if ( !reached[w] ) dfs(G,w,reached,count); 

} 

} 


bool Is_Connected(const graphs G) 

{ 

node_array <bool> reached(G, false); 

int count =0; 

node s = G.first_node(); 

if (s != nil) 

dfs(G,s,reached,count); 

return count == G.number_of_nodes(); 

} 


void Make_Connected(graphs G, list<edge>S L) 

{ 

node_array <bool> reached(G, false); 
node u = G.first_node() ; 
int count = 0; 


73 


node v; 

forall_nodes(V, G) 
if ( !reached[v] ) 

{ dfs(G, V, reached,count); // explore connected comp with root v 

if (u != v) // link v’s comp to the first comp 

L.append{G.new_edge{u, v)); 

} 

} 


list<edge> Make_Connected(graphs G) 

{ list<edge> L; 

if (G.number_of_nodes {) > 0) Make_Connected(G, L); 
return L; 

} 


static void make_bicon_dfs (graphs G, node v, ints dfs_count, 

list<edge>S L, 
node_array<int>S dfsnum, 
node_array<int>S lowpt, 
node_array<node>S parent) 

{ node u = nil; 

dfsnum[v] = df s__count++; 
lowpt[v] = dfsnum[v]; 

edge e; 

forall_inout_edges(e,v) 

{ 

node w = G.opposite(v, e) ; 

if (v == w) continue; // ignore loops 

if (u == nil) u = w; // first child 

if ( dfsnum[w] == -1) // w not reached before; e is a tree edge 
{ 

parent[w] = v; 

make_bicon_dfs(G, w, dfs_count, L, dfsnum, lowpt, parent); 
if (lowpt[w] == dfsnum[v]) 

{ // |v| is an articulation point. We now add an edge. If [w] is 

the 

// first child and |v| has a parent then we connect |w| and 
// |parent[v]|, if |w| is a first child and |v| has no parent 

then 

// we do nothing. If |w| is not the first child then we connect 

I w| 

// to the first child. The net effect of all of this is to link 

all 

// children of an articulation point to the first child and the 

first 

// child to the parent (if it exists) 
if (w == u SS parent[v]) 
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{ L.append(G.new_edge(w, parent[v])); 

//L.append(G.new_edge(parent[v], w)); (if bidirected) 

} 

if (w != u) 

{ L.append(G.new_edge(u, w)); 

//L.append(G.new_edge(w, u)); (if bidirected) 

} 

} 

lowpt[v] = Min(lowpt[v], lowpt[w]); 

} 

else // non tree edge 
lowpt [v] = Min (lowpt [v] , dfsnuin[w]); 


} 


} 


static bool is_bicon_dfs(const graphs G, node v, int& dfs_count, 

node_array<int>& dfsnum, 
node_array<int>& lowpt, 
node_array<node>& parent) 

{ node u = nil; 
edge e/ 

dfsnum[v] = dfs_count++; 
lowpt [v] = dfsnuin[v]; 

forall__inout_edges (e, v) 

{ node w = G.opposite(v,e); 
if (u == nil) u = w; 
if ( dfsnum[w] == -1 ) 

{ parent[w] = v; 

if (!is_bicon_dfs(G, w, dfs_count, dfsnum, lowpt, parent)) 
return false; 

if (lowpt [w] == dfsnuin[v] && (w != u 1| parent [v] ) ) return 

false; 

lowpt[v] = Min(lowpt[v], lowpt[w]); 


else 


lowpt[v] 


Min(lowpt[v], dfsnum[w]); 


return true; 

} 


bool Is_Biconnected(const graph & G) 

{ if (G.emptyO) return true; 

if ( ! Is_Connected(G) ) return false; 

node_array<int> lowpt(G) ; 
node_array<int> dfsnum(G,^1); 
node_array<node> parent(G, nil); 
int dfs_count =0; 

return is_bicon_dfs(G, G.first_node() , dfs_count, dfsnum, lowpt, 
parent); 

} 
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void Make_Biconnected(graphs G, list<edge>& L) 

{ 

if (G. nuiTiber_of_nodes () == 0) return; 

Make_Connected(G,L); 

node_array<int> lowpt(G); 

node_array<int> df snum (G,-1) ; // dfsnuin[v] == -1 <=> v not reached 

node_array<node> parent(G^nil); 

int dfs_count = 0; 

make_bicon_dfs(G, G.first_node(), dfs_count, L, dfsnum, lowpt, 
parent); 

} 


list<edge> Make_Biconnected(graph & G) 
{ list<edge> L; 

Make_Biconnected(G,L); 
return L; 

} 


static bool bi_bfs(const graphs G, node s, node_array<int>S side) 

{ 

list<node> Q; 

Q.append(s); 
side[s] = 0; 

while ( ! Q.empty 0 ) 

{ node V = Q.head(); 
edge e; 

forall_inout_edges(e,v) 

{ node w = G.opposite(v,e); 

if (side[v] == side[w]) return false; 
if (side[w] == -1) 

{ Q.append(w); 

side[w] = 1 - side[v]; 

) 

} 

Q.pop(); 

} 

return true; 

} 


bool Is_Bipartite(const graphs G, list<node>S A, list<node>S B) 

{ 

node_array<int> side(G,-l); 
node v; 
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} 


forall_nodes(v, G) 
if (side[v] == -1) 

if (! bi_bfs(G,V,side)) return false; 
forall_nodes(v, G) 

{ if (side[v] == 0) A.append(v); 
if (side[v] == 1) B.append(v); 

} 

return true; 


bool Is_Bipartite(const graphs G) 
{ list<node> A,B; 

return Is_Bipartite(G,A,B); 

} 


int COMPONENTS(const graph&G, node_array<int>&); 

int Genus(const graphs G) 

{ int n = G. nuinber_of_nodes () ; 

int m = G. nuiTiber_of_edges ()/2; // G is bidirected 

edge_array<bool> considered(G,false); 
int f = 0; 
edge e; 

forall_edges(e, G) 

{ if (G.reversal(e) == nil) 

error_handler(1,"Genus: graph must be a map."); 
if ( !considered[e] ) 

{ // trace the face to the left of e 
f++; 

edge x = e; 

do { considered[x] = true; 

X = G.face_cycle_succ(x); 

) 

while (x != e); 

} 

) 

node_array<int> cnum(G); 
int c = COMPONENTS(G,cnum)-1; 

return (2-n+m-f+c)/2; 

} 


/* 

void 

{ 


copy_graph(const graphs G, GRAPH<node,edge>S 

node_array<node>S v_in_H, edge_array<edge>S e_in_H) 


node v; 

forall_nodes(V,G) v_in_H[v] = H.new_node(v); 
edge e; 
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forall_edges(e,G) e_in_H[e] = 

H.new_edge(v_in_H[source(e)],v_in_H[target(e)],e); 

} 


bool Is_Map(const graphs G) 

{ edge_array<edge> rev(G); 

if (IIs_Bidirected(G,rev)) return false; 
edge x; 

forall_edges(x,G) 

{ edge y = G.reversal(x); 

if (x != G.reversal(y)) return false; 

if (source(x) != target(y) || source(y) != target (x)) return false 

} 

return true; 


bool Is_Planar_Map(const graphs G) { return Is_Map(G) SSGenus(G) == 0 

} 

bool Is_Planar(const graphs G) 

{ graph G1 = G; 

return PLANAR(G1); 

} 
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■k-k'k-fc'k'k-ic'k 

+ 

+ LEDA 3.5.1 
+ 

+ _g_obj ect s.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

*-k*-k-k**-k-k*-k-k**'k-k-k-k'k'k'k'k'k'k'k-k'k*'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k-k-k’k'k-k'tc'k-)c')c-k')c'k-k'kr'k-k*-k-k-Ar-k-k-k 

tinclude <LEDA/graph.h> 


// 


// nodes and edges 

// 

// by S. Naeher (1995) 
//- 


node_struct::node_struct(GenPtr inf) 

{ data[0] = inf; 
owner = nil; 
id = 0; 

for(int j=0; j<2; j++) 

{ first_adj_edge[j]= nil; 
last_adj_edge[j] = nil; 
adj_length[j] = 0; 

} 

} 

edge_struct::edge_struct(node v, node w, GenPtr i) 
{ succ_adj_edge[0] = nil; 
succ_adj_edge[1] = nil; 
pred_adj_edge[0] = nil; 
pred_adj_edge[1] = nil; 
id = 0; 
term[0] = v; 
term[l] = w; 
rev = nil; 
data[0] = i; 


face_struct;:face_struct(GenPtr x) 
{ data[0] = X ; 
id = 0; 
owner = nil; 
head = nil; 
sz = 0; 
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} 


void node_struct::append_adj_edge(edge e,int i, int chain_e) 

{ // append e to adj_list[i] 

// use succ/pred_adj_edge[chain_e] pointers for chaining of e 

edge last = last_adj_edge[i] ; 

e->succ_adj_edge[chain_e] = nil; 

if (last == 0) // empty list 
{ first_adj_edge[i] = e; 

e->pred_adj_edge[chain_e] = nil; 

} 

else 

{ e->pred_adj_edge[chain_e] = last; 

if (source(last) == target(last)) // loop 

last->succ_adj_edge[chain_e] = e; 
else 

last->succ_adj_edge[(this==source(last)) ? 0:1] = e; 

} 


last_adj_edge[i] = e; 
adj_length[i]++; 


} 


void node_struct::insert_adj_edge(edge e, edge el, int i, int chain_e, 
int dir) 

{ 

// insert e after (dir==0) or before (dir!=0) el into adj_list[i] 

// use succ/pred_adj_edge[chain_e] pointers for chaining 


if (el == nil) 

{ append_adj_edge (e, i, chain_e) ; 
return; 

} 


edge e2; // successor (dir==0) or predecessor (dir!=0) of el 

int chain_el; // chaining used for el 

if (source(el) == target(el)) // el is a self-loop 
chain_el = chain_e; 

else 

chain_el = (this == source(el)) ? 0 : 1; 
if (dir == 0) 

{ e2 = el->succ_adj_edge[chain_el]; 
e->pred_adj_edge[chain_e] = el; 
e->succ_adj_edge[chain_e] = e2; 
el->succ_adj_edge[chain_el] = e; 
if (e2 == nil) 

last_adj_edge[i] = e; 
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else 

{ if (source(e2) == target(e2)) //loop 
e2->pred_adj_edge[chain_e] = e; 
else 

e2->pred_adj_edge[(this==source(e2)) ? 0:1] = e; 

} 

} 

else 

{ e2 = el->pred_adj_edge[chain_el]; 
e->succ_adj_edge[chain_e] = el; 
e->pred_adj_edge[chain_e] = e2; 
el->pred_adj_edge[chain_el] = e; 
if (e2 == nil) 

first_adj_edge[i] = e; 
else 

{ if (source(e2) == target(e2)) //loop 
e2->succ_adj_edge[chain_e] = e; 
else 

e2->succ_adj_edge[(this==source(e2)) ? 0:1] = e; 

} 

} 

adj_length[i]++; 


void node_struct::del_adj_edge(edge e, int i, int chain_e) 

{ 

// remove e from adj_list[i] 

// with respect to succ/pred_adj_edge[chain_e] pointers 

edge e_succ = e->succ_adj_edge[chain_e]; 
edge e_pred = e->pred_adj_edge[chain_e]; 

if (e_succ) 

if (source(e_succ) == target(e_succ)) // loop 

e_succ->pred_adj_edge[chain_e] = e_pred; 
else 

e_succ->pred_adj_edge[(this==source(e_succ)) ? 0:1 ] 

else 

last_adj_edge[i] = e_pred; 
if (e_pred) 

if (source(e_pred) == target(e_pred)) // loop 
e_pred->succ_adj_edge[chain_e] = e_succ; 
else 

e_pred->succ_adj_edge[(this==source(e_pred)) ? 0:1 ] 

else 

first_adj_edge[i] = e_succ; 
adj_length[i]; 

} 


void graph_obj_list::clear() 
{ obj_list_head = 0; 


= e_pred 


= e succ 
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obj_list_tail = 0; 
obj_list_sz = 0; 


void graph_obj_liSt::append{graph_object* e) 

{ 

e->obj_list_succ = 0; 

if {obj_list_sz > 0) 

obj_list_tail->obj_list_succ = e; 
else 

obj_list_head = e; 

e->obj_list_pred = obj_list_tail; 
obj_list_tail = e; 

obj_list_sz++; 

} 


graph_obj ect* graph_obj_1ist::pop() 

{ graph_object* e = obj_list_head; 
if (e) 

{ graph_object* s = e->obj_list_succ; 
obj_list_head = s; 
if (s) 

S“>ob j_list_pred = 0; 
else 

obj_list_tail = 0; 
obj_list_sz-- ; 

} 

return e; 

} 


void graph_obj_list:: remove {graph_object* e) 

{ 

graph_obj ect * s = e-'>ob j_list_succ; 
graph_object* p = e~>obj_list_pred; 

if (s) 

{ e->obj_list_succ = s->obj_list_succ; 
s->obj_list_pred = p; } 

else 

obj_list_tail = p; 


if (p) 

{ e“>obj_list_pred = p->obj_list_pred; 
p->obj_list_succ = s; } 

else 

obj_list_head = s; 
obj_list_sz—; 


void graph_obj_list::cone{graph_obj_list& L) 
{ if {L.obj_list_sz ==0) return; 

if {obj_list_sz > 0) 
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obj_list_tail-‘>obj_list_succ = L. obj_list_head; 
else 

obj_list_head = L.obj_list_head; 

L.obj_list_head->obj_list_pred = obj_list_tail; 
obj_list_tail = L.obj_list_tail; 
obj_list_sz += L.obj_list_sz; 

L.clear(); 
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^-k-k-k-k-k-k-k-k-k-k-k'k-k-k-ic-k-k-k-k-k-fc-k-k-k-k-k-k-k-ffk-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-ic-k-k-k-k-k-k-k-k-k-k-k-k-k-k 

-k-k-k-k-k-k-k-k 

+ 

+ LEDA 3.5.1 
+ 

+ _g_partition.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104) . 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

■k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k 
■k -k -k -k -k -k -k ^ 

#include <LEDA/node_partition.h> 

void node_partition::init(const graphs G) 

{ node v; 

forall_nodes(V,G) ITEM[v] = partition::make_block(v)/ 

} 
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■Ar'Ar'*-+-Ar'*-'Ar + 

+ 

+ LEDA 3.5.1 
+ 

+ _g_random.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
t use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

#include <LEDA/graph.h> 

#include <LEDA/ugraph.h> 

// - 


// generators for random graphs 


// 

// S. Naeher 
//- 


// we use the global random integer source "rand^int" 

void random_graph(graphs G, int n, int m) 

{ /* random graph with n nodes and m edges */ 

node* V = new node[n]; 
int* deg = new int[n]; 
int i; 

G.clear (); 

for(i=0; i<n; i++) 

{ V [ i] = G.new_node(); 
deg[i] = 0; 

} 


for(i=0; i<m; i++) deg[rand_int(0,n-1)]++; 

for(i=0; i<n; i++) 

{ node V = V[i] ; 
int d = deg[i]; 

while (d--) G.new_edge(v,V[rand_int(0,n-1)]); 

} 

delete[] V; 
delete[] deg; 


void random_ugraph(ugraphS G, int n, int m) 
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{ int i; 

node* V = new node[n]; 

G.clear(); 

for(i=0;i<n;i++) V[i] = G.new_node(); 

while (m—) G.new_edge(V[rand_int(0,n-1)],V[rand_int(0,n-1)]); 

} 


void random_bigraph(graphs G,int nl,int n2,int m,list<node>& 
A, list<node>& B) 

{ 

int d = m/nl; 
int r = in%nl; 


node* AV = new node[nl]; 
node* BV = new node[n2]; 

A. clear(); 

B. clear(); 

G.clear(); 

for (int a = 0; a < nl; a++) A. append (AV [a] = G. new__node () ) ; 
for(int b = 0; b < n2; b++) B.append(BV[b] = G.new_node()); 

node v; 
int i; 

forall(v,A) 

for(i=0;i<d/i++) 

G. new_edge(v,BV[rand_int(0,n2-l)] ); 
while (r--) G,new_edge(AV[rand_int(0,nl-1)], BV[rand_int(0,n2-l)]); 


delete[] AV; 
delete[] BV; 


// 


// random planar graph 
//- 


#include <LEDA/sortseq,h> 
#include <LEDA/prio.h> 
#include <math.h> 

#define YNIL seq_item(nil) 
#define XNIL pq_item(nil) 


#define EPS 0,00001 
#define EPS2 0.0000000001 

class POINT; 
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class SEGMENT; 
typedef POINT* point; 
typedef SEGMENT* segment; 

enum point_type {lntersection=0,Rightend=l,Leftend=2}; 

class POINT 

{ 

friend class SEGMENT; 

segment seg; 
int kind; 

double x; 
double y; 

public: 

POINT(double a,double b) 

{ 

x=a; y=b; seg=0; kind=Intersection; 

} 


LEDA MEMORY(POINT); 


friend 

friend 

friend 

friend 


double get_x (point p) 
double get_y (point p) 
int get_kind(point 
segment get_seg(point p) 


P) 


{ return 
{ return 
{ return 
return p 


p->x; } 

p->y; } 
p->kind; 
>seg; } 


} 


friend bool intersection(segment, segment, points); 

}; 


static int compare(const points pi, const points p2) 
{ if (pl==p2) return 0; 

double diffx = get_x(pl) - get_x(p2); 
if (diffx > EPS2 ) return 1; 

if (diffx < -EPS2 ) return -1; 

int diffk = get_kind(pi)“get_kind(p2); 
if (diffk != 0) return diffk; 

double diffy = get_y(pl) - get_y(p2); 
if (diffy > EPS2 ) return 1; 

if (diffy < -EPS2 ) return -1; 

return 0; 


class SEGMENT 

{ 

point startpoint; 
point endpoint; 
double slope; 
double yshift; 
node left node; 
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int orient; 
int color; 
int name; 


public: 

SEGMENT(point, point,int,int); 

--SEGMENT () { delete startpoint; delete endpoint; } 

LEDA MEMORY(SEGMENT); 


} 


friend point get_startpoint(segment seg) 


{ return seg“>startpoint; 


friend point get_endpoint(segment seg) 
friend double get_slope(segment seg) 
friend double get_yshift(segment seg) 
friend node get_left_node(segment seg) 

>left_node; } 

friend void set_left_node(segment seg, node v) { seg->left node = v; } 


{ return seg->endpoint; } 

{ return seg~>slope; } 

{ return seg->yshift; } 
{ return seg- 


friend bool intersection(segment, segment, points); 


SEGMENT::SEGMENT(point pi,point p2,int c, int n) 

{ 

left_node = nil; 
color = c; 

name = n; 

if (compare(pi,p2) < 0) 

{ startpoint = pi; 
endpoint = p2; 
orient = 0; 

} 

else 

{ startpoint = p2; 
endpoint = pi; 
orient = 1; 

} 

startpoint“>kind = Leftend; 
endpoint“>kind = Rightend; 
startpoint“>seg = this; 
endpoint->seg = this; 

if (endpoint->x != startpoint->x) 

{ 

slope = (endpoint->y - startpoint->y)/(endpoint->x - startpoint- 

>x) ; 

yshift = startpoint->y - slope * startpoint->x; 

startpoint->x -= EPS; 
startpoint“>y -= EPS * slope; 
endpoint~>x += EPS; 
endpoint“>y += EPS * slope; 
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} 

else //vertical segment 
{ startpoint->y -= EPS; 
endpoint->y += EPS; 
slope = 0; 
yshift = 0; 


} 


static double x_sweep; 
static double y_sweep; 


static int compare(const segments si, const segments s2) 

{ 

double yl = get_slope (si) *x__sweep+get__yshift (si) ; 
double y2 = get_slope(s2)*x_sweep+get_yshift(s2); 

double diff = yl~y2; 
if (diff > EPS2 ) return 1; 
if (diff < -EPS2 ) return -1; 

if (get_slope(si) == get_slope(s2)) 

return compare(get_x(get_startpoint(si)), 
get_x(get_startpoint(s2))); 

if (yl <= y_sweep+EPS2) 

return compare(get_slope(si),get_slope(s2)) ; 

else 

return compare(get_slope(s2),get_slope(si)); 


} 


static priority_queue<seq_item, point> X__structure; 
static sortseq<segment,pq_item> Y_structure; 


bool intersection(segment segl,segment seg2, points inter) 

{ 

if (segl->slope == seg2->slope) 
return false; 
else 
{ 

double cx = (seg2->yshift - segl->yshift) / (segl->slope - seg2- 
>slope); 

if (cx <= x_sweep) return false; 

if (segl->startpoint->x > cx || seg2~>startpoint->x > cx || 

segl->endpoint~>x < cx || seg2->endpoint->x < cx ) return false; 

inter = new POINT(cx,segl->slope * cx + segl->yshift); 

return true; 

} 

} 
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inline pq_item Xinsert{seq_item i, point p) 
{ return X_structure.insert(i,p); } 


inline point Xdelete(pq_item i) 
{ point p = X_structure.inf(i); 
X_structure.del_item(i) ; 
return p; 

} 


void random_planar_graph(graphs G, node_array<double>& xcoord, 

node_array<double>& ycoord, int n) 

{ 

point p,inter; 

segment seg, 1,Isit,Ipred,Isucc,Ipredpred; 
pq_item pqit,pxmin; 

seq_item sitmin,sit,sitpred,sitsucc,sitpredpred; 

int MAX_X = n; 
int MAX Y = n; 


int N = n; // number of random segments 

G.clear(); 

Xcoord.init(G,n,0); 
ycoord.init(G, n, 0) / 


int count=l; 

//initialization of the X-structure 
for (int i = 0; i < N; i++) 

{ //point p = new POINT(rand_int(0,MAX_X/3), rand_int(0,MAX_Y)); 
//point q = new POINT (rand_int (2'*^MAX_X/3, MAX_X) , 
rand_int(0,MAX_Y)); 

point p = new POINT(rand_int(0,MAX_X), rand_int(0,MAX_Y)); 
point q = new POINT(rand_int(0,MAX_X) , rand_int(0,MAX_Y))/ 
seg = new SEGMENT(p,q,0,count++); 

Xinsert(YNIL,get_startpoint(seg)); 

} 

x_sweep = -MAXINT; 
y_sweep = -MAXINT; 

while( !X_structure.empty() && G.number_of_nodes() < n ) 

{ 

pxmin = X_structure.find_min(); 
p = X_structure.inf(pxmin); 

sitmin = X_structure.key(pxmin); 

Xdelete(pxmin); 

if (sitmin == YNIL) //left endpoint 
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1 = get_seg(p); 
x_sweep = get_x(p); 
y_sweep = get_y(p); 

node w = G.new_node() ; 
xcoord[w] = x_sweep; 
ycoord[w] = y_sweep; 
set_left_node(1,w); 

sit = Y_structure.insert(1,XNIL); 

Xinsert(sit,get_endpoint(1)); 

sitpred = Y_structure.pred(sit); 
sitsucc = Y_structure.succ(sit); 

if (sitpred != YNIL) 

{ if ((pqit = Y_structure.inf(sitpred)) != XNIL) 

delete Xdelete(pqit); 

Ipred = Y_structure.key(sitpred); 

Y_structure.change^inf(sitpred,XNIL); 

if (intersection(Ipred, 1,inter)) 

Y_structure.change_inf(sitpred,Xinsert(sitpred,inter)); 

} 


if (sitsucc != YNIL) 

{ Isucc = Y_structure.key(sitsucc); 
if (intersection(Isucc, 1, inter)) 

Y_structure.change_inf(sit,Xinsert(sit,inter)); 

} 


else if (get_kind(p) == Rightend) 

//right endpoint 

{ 

x_sweep = get_x(p); 
y_sweep = get_y(p); 

sit = sitmin; 

sitpred = Y_structure.pred(sit); 
sitsucc = Y_structure.succ(sit); 

segment seg = Y_structure.key(sit); 

Y^structure.del_item(sit); 

delete seg; 

if((sitpred != YNIL)&&(sitsucc != YNIL)) 

{ 

Ipred = Y_structure.key(sitpred); 

Isucc = Y_structure.key(sitsucc); 
if (intersection(Isucc,Ipred,inter)) 

Y_structure.change_inf(sitpred,Xinsert(sitpred,inter)) 

} 
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} 

else // intersection point p 

{ 

node w = G.new_node() ; 
xcoord[w] = get_x(p); 
ycoord[w] = get_y(p); 

/* Let L = list of all lines intersecting in p 

we compute sit = L.headO; 

and sitpred = L.tailO; 

by scanning the Y_structure in both directions 
starting at sitmin; 

V 

/* search for sitpred upwards from sitmin: ■*■/ 
Y_structure.change_inf(sitmin,XNIL); 
sitpred = Y_structure.succ(sitmin); 


while ((pqit=Y_structure.inf(sitpred)) != XNIL) 

{ point q = X_structure.inf(pqit); 
if (compare(p,q) != 0) break; 

X_structure.del_item(pqit); 

Y_structure.change_inf(sitpred,XNIL); 
sitpred = Y_structure.succ(sitpred); 

} 


/* search for sit downwards from sitmin: */ 
sit = sitmin; 
seq_item sitl; 

while ((sitl=Y_structure.pred(sit)) 1= YNIL) 

{ pqit = Y_structure.inf(sitl); 
if (pqit == XNIL) break; 
point q = X_structure.inf(pqit); 
if (compare(p,q) != 0) break; 

X_structure.del_item(pqit); 

Y_structure.change_inf(sitl,XNIL); 
sit = sitl; 

} 


// insert edges to p for all segments in sit, ..., 

into G 

// and set left node to w 

Isit = Y_structure.key(sitpred); 

node V = get_left_node(Isit); 
if (v!=nil && w!=nil) G.new_edge(v,w); 
set left node(Isit,w); 


sitpred 
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for(sitl=sit; sit1!=sitpred; sitl = Y_structure.succ(sitl)) 
{ Isit = Y_structure.key(sitl); 

V = get_left_node(Isit); 

if (v!=nil && w!=nil) G.new_edge(v,w); 

set_left_node(Isit,w); 

} 

Isit = Y_structure.key(sit); 
lpred=Y_structure.key(sitpred); 
sitpredpred = Y_structure.pred(sit) ; 
sitsucc=Y_structure.succ(sitpred); 


if (sitpredpred != YNIL) 

{ 

lpredpred=Y_structure.key(sitpredpred); 

if ((pqit = Y_structure.inf(sitpredpred)) != XNIL) 

delete Xdelete(pqit); 

Y_structure.change_inf(sitpredpred,XNIL); 


if (intersection(Ipred,Ipredpred,inter)) 

Y_structure.change_inf(sitpredpred, 

Xinsert(sitpredpred,inter)) 


} 


if (sitsucc != YNIL) 

{ 

lsucc=Y_structure.key(sitsucc); 

if ((pqit = Y_structure.inf(sitpred)) != XNIL) 

delete Xdelete(pqit); 

Y_structure.change_inf(sitpred,XNIL); 

if (intersection(Isucc,Isit,inter)) 

Y_structure.change_inf(sit,Xinsert(sit,inter)); 

} 


// reverse the subsequence sit, ... ,sitpred in the Y_structure 

x_sweep = get_x(p); 
y_sweep = get_y(p); 

Y_structure.reverse_iterns(sit,sitpred) ; 
delete p; 

} // intersection 


pq_item xit; 

forall_items(xit,X_structure) 

{ point p = X_structure.inf(xit); 
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seq_itein sit = X_structure . key (xit) ; 

if (get_kind(p) == Leftend) delete get_seg(p); 

if (get_kind(p) == Rightend && sit) delete Y_structure.key(sit); 

if (get_kind(p) == Intersection) delete p; 

} 

X_structure.clear(); 

Y_structure.clear(); 

Make_Connected(G); 

// normalize x and y coordinates 
node v; 

forall_nodes(v, G) 

{ xcoord[v] /= x^sweep; 
ycoord[v] /= n; 

} 


void random_planar_graph(graphs G, int n) 
{ node_array<double> xcoord; 
node_array<double> ycoord; 
random_planar_graph(G,xcoord,ycoord,n)/ 
//random_planar_graph(G, n, n) ; 
//Make_Connected(G) ; 

} 


void maximal_planar_map(graphs G, int n) 

{ 

G.clear(); 

if (n <= 0 ) return; 

node a = G.new_node()/ 
n—; 

if (n == 0) return; 

node b = G.new^node(); 
n—; 


edge* E = new edge[6*n]; 

E[0] = G.new^edge(a,b); 

E[l] = G.new_edge(b,a); 

G.set_reversal(E[0],E[1]); 

int m = 2; 

while (n—) 

{ edge e = E[rand_int(0,m-1)]; 
node V = G.new_node(); 
while (target(e) != v) 

{ edge x = G.new_edge(v, source(e)) ; 
edge y = G.new_edge(e,v,after); 
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E[m++] = x; 

E[in++] = y; 

G. set_reversal (x, y) ; 
e = G.face_cycle_succ(e); 

} 


delete[] E; 

} 


void maximal_planar_graph(graphs G, int n) 

{ 

inaximal_planar_map (G, n) ; 
list<edge> E; 

edge_array<bool> marked(G,false); 
edge e; 

forall_edges(e,G) 

{ if (!marked[e]) E.append(e); 
marked[e] = true; 
marked[G.reversal(e)] = true; 

} 

fora11(e,E) G.del_edge(e); 

} 


void random_planar_map(graphs G, int n, int m) 

{ 

maximal_planar_map(G,n); 
list<edge> E; 

edge_array<bool> marked(G,false) ; 
edge e; 

forall_edges(e,G) 

{ if (!marked[e]) E.append(e); 
marked[e] = true; 
marked[G.reversal(e)] = true; 

} 


E.permute(); 

while (E.lengthO > m) 

{ edge e = E.pop(); 

edge r = G.reversal(e); 
G.del_edge(e) ; 
G.del_edge(r) ; 

} 


} 


void random_planar_graph (graphs G, int n, int m) 

{ 

random_planar_map(G,n,m); 
list<edge> E; 
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edge_array<bool> marked(G,false) 
edge e; 

forall_edges(e,G) 

{ if (!marked[e]) E.append(e); 
marked[e] = true; 
marked[G.reversal(e)] = true; 

} 

forall(e,E) G.del_edge(e); 


★ ★★★★★★■A- 
+ 

+ LEDA 3.5.1 
+ 

+ _g_sort.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 

+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

#include <LEDA/graph.h> 

// - 

// sorting 

// 

// by S. Naeher (1995) 

// - 


static const graph^map* NA; 
static graph* GGG; 


/* 

static int array_cmp_nodes(const nodes x, const nodes y) 

{ return NA->cmp_entry(NA->array_read(x),NA->array_read(y)) ; } 

static int array_cmp_edges(const edges x, const edges y) 

{ return NA->cmp_entry (NA->array_read(x),NA->array_read(y) ) ; } 

*/ 

static int int_array_cmp_nodes(const nodes x, const nodes y) 

{ return LEDA_COMPARE (int, NA->array_read (x) , NA->array_read (y) ) ; } 

static int int_array_cmp_edges(const edges x, const edges y) 

{ return LEDA_COMPARE (int, NA->array_read (x) , NA->array_read (y) ) ; } 

static int float_array_cmp_nodes(const nodes x, const nodes y) 

{ return LEDA_COMPARE (float, NA->array_read (x) , NA->array_read (y) ) ; } 

static int float_array_cmp_edges(const edges x, const edges y) 

{ return LEDA_COMPARE ( float,NA->array_read(x),NA->array_read(y) ) ; } 

static int double_array_cmp_nodes (const nodes x, const nodes y) 

{ return LEDA_COMPARE (double, NA->array_read (x) , NA->array_read (y) ); } 

static int double_array_cmp_edges (const edges x, const edges y) 

{ return LEDA_COMPARE (double, NA->array_read (x) , NA->array_read (y) ) ; } 
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static int graph_cmp_nodes(const node& x, const node& y) 
{ return GGG->cmp_node_entry(x, y) ; } 

static int graph_cmp__edges (const edges x, const edges y) 
{ return GGG->cmp_edge_entry(x, y) ; } 


void graph;:sort_nodes(const list<node>S vl) 

{ 

if (vl.length 0 != number_of_nodes()) 

error_handler(1,"graph::sort_nodes(list<node>); illegal node 

list") ; 

v_list.clear (); 

node v; 
forall(V,vl) 

{ if (v->owner != this) 

error_handler(1,"graphsort_nodes(list<node>): illegal node 

list") ; 

v_list.append(v); 

} 

} 


void graphsort_nodes(int (*f)(const nodes, const nodes)) 
{ list<node> vl = all_nodes(); 
vl.sort(f); 
sort_nodes(vl); 


void graph::sort_edges(const list<edge>S el) 

{ 

node v; 
edge e; 

if (el. length 0 != nuitiber_of_edges () ) 

error_handler(1,"graph::sort_edges(list<edge>): illegal edge 
list") ; 


// clear all adjacency lists 
forall_nodes(v,*this) 
for(int i=0; i<2; i++) 

{ v->first_adj_edge[i] = 0; 
v->last_adj_edge[i] = 0; 
v->adj_length[i] =0; 

} 


e_list.clear(); 

forall(e,el) 

{ 

if (e->term[0]->owner != this) 

error_handler(1,"graph::sort^edges(list<edge>): edge not in 

graph"); 
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e_list.append(e); 


source (e) ’->append_adj_edge (e, 0, 0) / 
if (undirected) 

target(e)->append_adj_edge(e,0,1); 
else 

target(e)->append_adj_edge(e,1); 

} 

} 


void graph::sort_edges(int (*f)(const edge&, const edge&)) 
{ list<edge> el = all_edges(); 
el.sort(f); 
sort_edges(el); 

} 


void graph::sort_nodes(const graph_map& A) 

{ NA = &A; 

switch (A.elem_type_id() ) { 

case INT_TYPE_ID; sort_nodes(int_array_cmp_nodes); 

break; 

case FLOAT_TYPE_ID: 

sort_nodes(float_array_cmp_nodes); 
break; 

case DOUBLE_TYPE_ID: 

sort_nodes (double__array_cmp_nodes) ; 
break; 

default: 

error_handler(1,”G.sort_nodes(node_array<T>): T must be 
numerical.”); 

} 

} 


void graph::sort_edges(const graph_map& A) 

{ NA = &A; 

switch (A.elem_type_id()) { 

case INT_TYPE_ID: sort_edges(int_array_cmp_edges); 

break; 

case FLOAT_TYPE_ID: 

sort_edges (float_array_cmp_edges) ; 
break; 

case DOUBLE_TYPE_ID: 

sort_edges(double_array_cmp_edges); 
break; 

default: 

error_handler(1,”G.sort_edges(node_array<T>): T must be 
numerical."); 

} 

} 


void graph::sort_nodes() 

{ GGG = this; 

sort_nodes(graph_cmp_nodes); 

} 
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void graphsort_edges() 

{ GGG = this; 

sort_edges(graph_cmp_edges); 

} 


// bucket sort 

static int array_ord_node(const node& x) 

{ return LEDA_ACCESS(int,NA->array_read(x)); ) 

static int array_ord_edge(const edge& x) 

{ return LEDA__ACCESS (int, NA->array_read (x) ) ; } 


void graph::bucket_sort_nodes(int 1, int h, int (^ord)(const node&)) 
{ list<node> vl = all_nodes(); 
vl.bucket_sort(1,h,ord); 
sort nodes(vl); 


void graph::bucket_sort_edges(int 1, int h, int (*ord)(const edge&)) 
{ list<edge> el = all_edges(); 
el.bucket_sort(1,h,ord); 
sort_edges(el); 

} 

void graph::bucket_sort_nodes(int (*ord)(const node&)) 

{ list<node> vl = all_nodes(); 
vl.bucket_sort(ord); 
sort nodes(vl); 


void graphbucket__sort_edges (int (*ord) (const edge&)) 
{ list<edge> el = all_edges(); 
el.bucket_sort(ord); 
sort_edges(el); 

} 


void graph::bucket_sort_nodes(const graph_map& A) 
{ NA = &A; 

switch (A.elem_type_id()) { 


case INT_TYPE_ID: bucket_sort_nodes(array_ord_node); 

break; 


default: 

error_handler(1,"G.bucket_sort_nodes(node_array<T>): 
integer."); 

} 

} 


T must be 


void graph::bucket_sort_edges(const graph_map& A) 

{ NA = &A; 

switch (A.elem_type_id()) { 

case INT_TYPE_ID: bucket_sort_edges(array_ord_edge); 
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break; 


default: 

error_handler (1, *'G . bucket_sort_edges (edge_array<T>) : T must be 
integer."); 

} 

} 
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+ 

+ LEDA 3.5.1 
+ 

+ _gml_graph.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104) . 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 


// - // 

// class gml_graph, parser for graphs in GML format // 
// // 
// by David Alberts (1997) // 
//- // 


#include<LEDA/gml_graph.h> 


void gml_graph::init_rules() 

{ 

// graph rule 

append("graph”); 

add_rule(new_graph,gml_list) ; 

// directed graph (default) or not? 
append("directed”); 
add_rule(directed,gml_int) ; 
goback(); 

// type of node parameter (optional) 
append("nodeType”)/ 
add_rule(nodeType,gml_string); 
goback(); 

// type of edge parameter (optional) 
append("edgeType"); 
add_rule(edgeType,gml_string) ; 
goback(); 

// new node 

append("node”)/ 

add_rule(new_node,gml_list); 

// node index 
append("id”); 

add_rule(node_index,gml_int); 
goback(); 

// node parameter 

append("parameter”); 

add_rule(node_param,gml^string) ; 
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goback(); 

goback(); 

// new edge 

append("edge" ) ; 

add_rule (new_edge, ginl_list) ; 

// edge source 
append("source"); 
add_rule(edge_source,gml_int); 
goback(); 

// edge target 
append("target"); 
add_rule(edge_target,gml_int); 
goback(); 

// edge parameter 

append("parameter"); 

add_rule(edge_param,gml_string)/ 

reset_path(); 

} 

void gml_graph::add_graph_rule(gml_graph_rule f, gml_value_type 
key) 

{ 

if(key) 

{ 

reset_path(); 
append("graph"); 
append(key); 

} 

graph_rules[next_rule] = f; 
add_rule(next_rule,t); 
next_rule++; 

if(key) reset_path(); 

} 


void gml_graph::add_node_rule(gml_node_rule f, gml_value_type t, 
key) 

{ 

if(key) 

{ 

reset_path() ; 
append("graph"); 
append("node"); 
append(key) ; 

} 


node_rules[next_rule] = f; 
add_rule(next_rule,t); 
next_rule++; 

if(key) reset_path(); 

} 


t, char* 


char* 


103 


void gml_graph: : add_edge_rule (gml_edge_rule t, gml_value_type t, char* 
key) 

{ 

if(key) 

{ 

reset_path(); 
append("graph"); 
append("edge"); 
append(key); 

} 

edge_rules[next_rule] = f; 
add_rule(next_rule,t); 
next_rule++; 

if(key) reset^path(); 

} 


bool gml_graph::interpret(gml_rule r, const gml_object* gobj) 

{ 

bool ok = true; 

switch(r) 

{ 

case new_graph: 

{ 

ok = graph_intro(gobj); 
break; 

} 

case directed: 

{ 

if(gobj“>get_int{)) the_graph“>make_directed(); 
else the_graph->make_undirected(); 

break; 

} 

case nodeType: 

{ 

right_node_type = ! strcmp (the_graph->node_type () , gobj- 
>get_string()); 
break; 

} 

case edgeType: 

{ 

right_edge_type = !strcmp(the_graph->edge_type(),gobj- 
>get_string()); 
break; 

} 

case new_node: 

{ 

current_node = the_graph->new_node() ; 
has_id = false; 
gml_node_rule f; 

forall(f,new_node_rules) ok = ok && 

(*f)(gobj,the_graph,current_node); 
break; 

} 

case node_index: 

{ 

(*(node_by_id))[gobj->get_int()] = current_node; 
has id = true; 
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break; 


} 

case node_param; 

{ 

the_graph->set_node_entry{current_node, string(gobj- 
>get_string())); 
break; 

} 

case new_edge: 

{ 

edge e = the_graph“>new_edge (dummy 1, duininy2) ; 

{*(edge_s))[e] = -1; 

(*(edge_t))[e] = ~1; 
current_edge = e; 
gml_edge_rule f; 

forall(f,new_edge_rules) ok = ok && 

()(gobj,the_graph,current_edge); 
break; 

} 

case edge_source: 

{ 

(*(edge_s))[current_edge] = gobj->get_int{); 
break; 

} 

case edge_target: 

{ 

(*(edge_t))[current_edge] = gobj->get_int(); 
break; 

} 

case edge_param: 

{ 

the_graph“>set_edge_entry{current_edge, string(gobj- 
>get_string())); 
break; 

} 

default: 

{ 

if(node_rules.defined(r) ) 

{ 

ok = (*(node_rules[r])) (gobj,the_graph, current_node) 
break; 

} 

if(edge_rules.defined(r)) 

{ 

ok = (*(edge_rules[r]))(gobj,the_graph,current_edge) 
break; 

} 

if (graph__rules . defined (r) ) 

{ 

ok = (*(graph_rules[r]))(gobj,the_graph); 
break; 

} 

break; 

} 


return ok; 

} 

bool gml_graph::list_end(gml_rule r, const gml_object* gobj) 

{ 
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bool ok = true; 


switch(r) 

{ 

case new_graph: 

{ 

ok = graph_end{gobj); 
break; 

} 

case new_node: 

{ 

if(!has_id) 

{ 

print_error(*gobj,"missing node id"); 
ok = false; 

} 

gml_node_rule f; 

forall{f,node_done_rules) ok = ok && 
{*f)(gobj,the_graph,current_node); 

current_node = nil; 

break; 

} 

case new_edge: 

{ 

ok = edge_end(gobj); 
break; 

} 

default: 

{ 

break; 


} 

return ok; 

} 


bool gml_graph::graph_intro(const gml_object* gobj) 

{ 

node_by_id = new map<int,node>; 
edge_s = new map<edge,int>; 
edge_t = new map<edge,int>; 

the_graph->clear{); 

dummyl = the_graph->new_node(); 

dummy2 = the_graph->new_node() ; 

right_node_type = true; 
right_edge_type = true; 

// call new graph rules 
bool ok = true; 
gml_graph_rule f; 

forall(f,new_graph_rules) ok = ok && (*f)(gobj,the_graph); 
return ok; 

} 
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bool gml_graph::graph_end(const gml_object* gobj) 

{ 

bool ok = true; 

if (! right__node__type || ! right_edge_type) 

{ 

if(!right_node_type) 

print_error(*gobj,"wrong node type"); 
if(!right_edge_type) 

print_error(*gobj,"wrong edge type"); 
ok = false; 


edge e; 
node s,t; 

// settle edges 
forall_edges(e,*the_graph) 

{ 

s = t = 0; 

if((*(edge_s))[e] != -1) 

s = (*(node_by_id))[(*(edge_s))[e]]; 
if((*(edge_t))[e] != -1) 

t = (* (node_by_id) ) [ (* (edge__t) ) [e] ] ; 
if(s && t) the_graph->move_edge(e, s, t); 


the_graph->del_node(dummyl); 
the_graph->del_node (duininy2) ; 

// call graph done rules 
gml_graph_rule f; 

forall(f,graph_done_rules) ok = ok && (*f)(gobj,the_graph) 

if (node__by_id) 

{ 

delete node_by_id; 
delete edge_s; 
delete edge^t; 

node_by_id = 0; 
edge_s = 0; 
edge_t = 0; 


return ok; 


bool ginl_graph: : edge_end (const gml_object* gob j ) 

{ 

bool ok = true; 

if((*(edge_s))[current_edge] == -1) 

{ 

print_error(*gobj,"edge without source"); 
ok = false; 

} 

if((*(edge_t))[current_edge] == -1) 

{ 

print_error(*gobj,"edge without target"); 
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ok = false; 


} 

ginl_edge_rule f; 

forall (f, edge__done_rules) ok = ok && 
(* f) (gobj , the_graph,current_edge); 

current_edge = nil; 

return ok; 

} 
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^'k'k'k-k-k-k’k-k'k-k-k-k-k-k-k'k'k-k-k'k'k'k'k'k'k'k'k'k'k'k-k-k'k-k-k-ic'ir'k'k'k'k'k'k-k-k-k-ic-k-k-ie-k-k'k-k-k-k-k-k'k'k'k-k'k-k-k-ic-k-ie'k-ic-k 

'k'k-k'ir^-k^-k 

+ 

+ LEDA 3.5.1 
+ 

+ _graph.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

#include <LEDA/graph.h> 

// - 


// basic graph operations 

// 

// by S. Naeher (1995,1996) 
//- 


graph::graph() 

{ int szl = LEDA::node_data_slots; 
int sz2 = LEDA::edge_data_slots; 
max_n_index = -1; 
max_e_index = -1; 
max_f_index = -1; 
parent = 0; 
undirected = false; 
data_sz[0] = szl; 
data_sz[l] = sz2; 
data_sz[2] = 0; 

while (szl) free_data[0].push(szl—); 
while (sz2) free_data[1].push(sz2--); 
node_data_map = new graph_map(this, 0); 
edge_data_map = new graph_map(this,1); 
adj_iterator = new graph_map(this,0,0); 
FaceOf = 0; 

} 


graph::graph(int szl, int sz2) 

{ max_n_index = -1; 
max_e_index = -1; 
max_f_index = -1; 
parent = 0; 
undirected = false; 
data__sz[0] = szl; 
data_sz[l) = sz2; 
data_sz[2] = 0; 

while (szl) free_data[0].push(szl--); 
while (sz2) free_data[l].push(sz2--); 
node_data_map = new graph_map(this,0); 
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edge_data_map = new graph_iuap {this, 1) ; 
adj_iterator = new graph_map(this,0,0); 
FaceOf = 0; 

} 


void graph::copy_all_entries() const 
{ node v; 

forall_nodes(v,*this) copy_node_entry{v->data[0]); 
edge e; 

forall_edges(e,*this) copy_edge_entry{e->data[0]); 

// hidden edges 

for{e = (edge)h_list.head 0; e; e = (edge)h_list.succ(e)) 
copy_edge_entry{e->data[0]); 

} 


void graph::clear_all_entries{) const 
{ node v; 

forall_nodes(V,*this) clear_node_entry{v->data[0]); 
edge e; 

forall_edges(e,*this) clear_edge_entry(e->data[0]); 

// hidden edges 

for{e = (edge)h_list.head(); e; e = (edge)h_list.succ(e)) 
clear_edge_entry(e->data[0]); 

} 


void graph::copy_graph(const graphs G) 

{ 

int n = G.number_of_nodes(); 

//int m = G.number_of_edges(); 

for(int k = 0; k < 3; k++) 

{ data_sz[k] = G.data_sz[k]; 

for(int i=l; i<=data_sz[k]; i++) free_data[k].append(i) 

} 

max_n_index = -1; 
max_e_index = -1; 
inax_f_index = -1; 

e_list.clear(); 

FaceOf = 0; 

parent = 0; 

if (n == 0) return; 

node* node_vec = new node[G.max_n_index+l]; 
edge* edge_vec = new edge[G.max_e_index+1]; 

if (node_vec == 0 || edge_vec == 0) 

error_handler (1, copy_graph: out of memory"); 

// allocate a single block of memory for all nodes 
// memory_allocate_block(sizeof(node_struct) , n) ; 

node v; 
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forall_nodes(v,G) 

node_vec[index(v)] = new_node(v->data[0]); 

// allocate a single block of memory for all edges 
// memory_allocate_block(sizeof(edge_struct),m); 

bool loops_deleted = false; 

forall_nodes(v,G) 

{ node a = node_vec[index(v)]; 
edge e; 

forall_adj_edges(e, v) 

{ if ( undirected && v == target(e)) //incoming edge 
{ if (v == source(e)) // loop 

loops_deleted = true; 

} 

else 

{ node b = node_vec[index(target(e))]; 

edge_vec[index(e)] = new_edge(a,b,e->data[0]); 

} 

} 

} 


// update reversal information 
edge e; 

forall_edges(e,G) 
if (e“>rev) 

edge_vec[index(e)]->rev = edge_vec[index(e“>rev)]; 


// copy faces (if existing) 
face f; 

forall_faces(f,G) 

{ face fl = new_face(f->data[0]); 

fl->head = edge_vec[index(f->head)]; 

} 

delete[] node_vec; 
delete[] edge_vec; 

if ( loops_deleted ) 

error_handler(0,"selfloops deleted in ugraph copy constructor"); 


} 


graphgraph(const graphs G) 

{ undirected = G.undirected; 
copy_graph(G); 

node_data_map = new graph_map(this,0); 
edge_data_map = new graph_map(this,1); 
adj_iterator = new graph_map(this,0,0); 

} 


graphs graph:: operator= (const graph*S G) 
{ if (SG != this) 

{ graph::clear(); 

undirected = G.undirected; 
copy_graph(G); 
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} 

return *this; 


void graph::join(graphs G) 

{ // moves all objects from G to this graph and clears G 

if (G.undirected != undirected) 

error_handler(1,"graph::join(G): cannot merge directed and 
undirected graphs."); 

for(int d=0; d<3; d++) { 

if (G.data_sz[d] != data_sz[d]) 

error_handler(1,"graph::join(G): cannot merge graphs with 
different data sizes."); 

} 

node V; 
edge e; 
face f; 

int i = max_n_index; 

forall_nodes(V,G) { v->id = ++i; v->owner = this; } 
max_n_index = i; 

int j = max_e_index; 
forall_edges(e,G) e“>id = ++j; 
max_e_index = j; 

int k = max_f_index; 
forall_faces(f,G) f“>id = ++k; 
max_f_index = k; 

v_list.cone(G.v_list) ; 
e_list.cone(G.e_list) ; 
f_list.cone(G.f_list) ; 

G.max_n_index = -1; 

G.max_e_index = -1; 

G, max_f__index = -1; 

} 


// subgraph constructors (do not work for undirected graphs) 

/* 

graph::graph(graphs G, const list<node>S nl, const list<edge>S el) 
{ // construct subgraph (nl,el) of graph G 


parent = SG; 
node v,w; 
edge e; 

node* N = new node[G.max_n_index+l] ; 
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forall(v,nl) 

{ if (graph_of(v) != parent) 

error_handler(1,"graph: illegal node in subgraph constructor"); 
N[index(v)] = new_node((GenPtr)v) ; 

} 

forall(e,el) 

{ V = source(e); 
w = target(e); 

if ( graph_of(e)!= parent || N[index(v)]==0 |1 N[index(w)]==0 ) 
error_handler(1,"graph: illegal edge in subgraph constructor"); 
new_edge(N[index(v)],N[index(w)],(GenPtr)e); 

} 

undirected = G.undirected; 
delete[] N; 


graph::graph(graphs G, const list<edge>& el) 

{ // construct subgraph of graph G with edge set el 

node v,w; 
edge e; 

node* N = new node[G.inax_n_index+l]; 

forall_nodes(V,G) N[index(v)] = 0; 

parent = &G; 

forall(e,el) 

{ V = source(e); 
w = target(e); 

if (N[index(v)] == 0) N[index(v)] = new_node((GenPtr)v); 
if (N [index (w)] == 0) N [index (w)] = new__node ( (GenPtr) w) ; 
if ( graph_of(e) != parent ) 

error_handler(1,"graph: illegal edge in subgraph constructor"); 
new_edge(N[index(v)],N[index(w )], (GenPtr)e); 

) 


undirected = G.undirected; 
delete[] N; 


) 

*/ 

//- 

// destruction 
//- 


void graph::del_all_nodes() { clear(); } 

void graph::del_all_edges() 

{ 
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edge e; 

e = (edge)e_list.head(); 
while (e) 

{ edge next = (edge) e__list. succ (e) ; 
dealloc_edge(e); 
e = next; 

} 


e = (edge)h_list.head 0; 
while (e) 

{ edge next = (edge)h_list.succ(e); 
dealloc_edge(e); 
e = next; 

} 


e = (edge)e_free.head 0 ; 
while (e) 

{ edge next = (edge)e_free.succ(e) ; 
dealloc_edge(e); 
e = next; 

} 


e_list.clear(); 
h_list.clear(); 
e_free.clear(); 

max_e__index = -1; 

node v; 

forall_nodes(v,*this) 
for(int i=0; i<2; i++) 

{ v->first_adj_edge[i] = nil; 
v->last__adj__edge [i] = nil; 
v->adj_length[i] = 0; 

} 


void graph;;del_all_faces() 

{ 

face f = (face) f_list .headO ; 
while (f) 

{ face next = (face)f_list.succ(f) ; 
dealloc__face (f) ; 
f = next; 

} 


f = (face)f_free.head 0 ; 
while (f) 

{ face next = (face)f_free.succ(f); 
dealloc_face(f); 
f = next; 

} 


f_free.clear(); 
f_list.clear(); 

if (FaceOf) 

{ delete FaceOf; 
FaceOf = 0; 
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} 


max_f_index = -1; 

} 


void graph::clear() 

{ 

pre_clear_handler() ; 

for(int k=0; k<3; k++) 

{ graph_map* m; 

forall(m,map_list[k]) 

if (m->g_index > 0) m->clear_table(); 

} 

del_all_faces(); 
del_all_edges(); 

node V = (node)v_list.head (); 
while (v) 

{ node next = (node)v_list.succ(v); 
dealloc_node(v)/ 

V = next; 

} 


V = (node)v_free.head(); 
while (v) 

{ node next = (node)v_free.succ(v); 
dealloc_node(v); 

V = next; 

} 


v_list.clear(); 
v_free.clear(); 

max_n_index = -1; 

post_clear_handler(); 


graph::-graph() 

{ clear(); 

for(int k=0; k<3; k++) 

{ graph_map* m; 

forall(m,map_list[k]) m->g = 0; 

} 

delete adj^iterator; 
delete node_data_map; 
delete edge_data_inap; 


// 


// accessing node and edge lists 
//- 
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const list<node>& graph::all_nodes() const 
{ ((list<node>&)v_list_tmp).clear(); 

node v; 

forall_nodes{v,*this) 

((list<node>&)v_list_tmp).append(v); 
return v_list_tmp; 

} 


const list<edge>& graph::all_edges() const 
{ ((list<edge>&)e_list_tmp).clear() / 

edge e; 

forall_edges(e,*this) 

{{list<edge>&)e_list_tmp) .append(e) ; 
return e_list_tmp; 

} 

const list<face>& graph::all_faces() const 
{ ((list<face>&)f_list_tmp).clear(); 

face f; 

forall_faces{f,*this) 

({list<face>&)f_list_tmp).append(f); 
return f_list_tmp; 

} 


list<edge> graph::out_edges(node v) const 
{ list<edge> result; 
edge e; 

forall_out_edges(e, v) result.append(e); 
return result; 


list<edge> graph::in_edges(node v) const 
{ list<edge> result; 
edge e; 

forall_in_edges(e, v) result.append(e); 
return result; 


list<edge> graph::adj_edges(node v) const 
{ list<edge> result; 
edge e; 

forall_adj_edges(e,v) result.append(e); 
return result; 


list<node> graph::adj_nodes(node v) const 
{ list<node> result; 
edge e; 

forall_adj_edges(e, v) result.append(opposite(v,e)); 
return result; 
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// 


// update operations 
//- 


list<edge> graph::insert_reverse_edges() 

{ list<edge> L; 

edge e = first_edge()/ 

if (e != nil) 

{ L.append(new_edge(target(e),source(e),e->data[0])); 
copy_edge_entry(e->data[0] ) ; 
e = succ_edge(e) ; 

} 

edge stop = last_edge(); 
while (e != stop) 

{ L.append(new_edge(target(e),source(e),e->data[0])); 
copy_edge_entry(e->data[0]) ; 
e = succ_edge(e) ; 

} 

return L; 

} 


face graph::add_face(GenPtr inf) 

{ face f; 

if ( f_free.empty() ) 

{ f = (face)std_memory.allocate_bytes(face_bytes()); 
new (f) face_struct(inf); 
f->owner = this; 
f->id = ++max_f_index; 

} 

else 

{ f = (face)f_free.pop 0; 
f->data[0] = inf; 

} 

f_list.append(f) ; 
graph_map* m; 

forall(m,map_list[2]) m->re_init_entry(f) ; 
return f; 

} 

void graph::dealloc_face(face f) 

{ std_memory.deallocate_bytes(f,face_bytes() ) ; } 

void graph::del_face(face f) 

{ f_list.remove(f) ; 
f_free.append(f) ; 
graph_map* m; 
forall(m,map_list[2]) 

{ int i = m->g_index; 

if (i > 0) m->clear_entry(f->data[i] ) ; 
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} 


} 


node graph::add_node(GenPtr inf) 

{ node v; 

if ( v_free.empty() ) 

{ V = (node)std_memory.allocate_bytes{node_bytes{)) 
new (v) node_struct(inf)/ 

V“>owner = this; 
v->id = ++max_n_index; 
v->succ_link = nil; 

} 

else 

{ V = (node)v_free.pop 0; 

V“>data[0] = inf; 

} 

v_list.append(v); 
graph_map* m; 

forall(m^map_list[0]) m“>re_init_entry(v); 
return v; 

} 


void graph::dealloc_node(node v) 

{ std_memory.deallocate_bytes(v,node_bytes()); } 


node graph::new_node() 

{ GenPtr x = 0; 

pre_new_node_handler(); 
init_node_entry(x); 
node V = add_node(x); 
post_new_node_handler(v); 
return v; 


node graph::new_node(GenPtr i) 
{ pre_new_node_handler(); 
node V - add_node(i); 
post_new_node_handler(v); 
return v; 


void graph::del_node(node v) 

{ 

if (v->owner != this) 

error handler(4,"del node(v): v is not in G"); 


// delete adjacent edges 
edge e; 

while ((e=v->first_adj_edge[0]) != nil) del_edge(e); 
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if (!undirected) 

while ( (e=v->first_adj__edge [ 1 ] ) != nil) del_edge(e); 

pre_del_node_handler (v) ; 

if (parent==0) clear_node_entry (v*'>data [0] ) ; 

v_list. remove (v); 
v_free.append(v); 

graph_map* m; 

fora11(m,map_list[0]) 

{ int i = m->g_index; 

if (i > 0) m->clear_entry(v->data[i] ) ; 

} 

post_del_node_handler(); 


node graph::merge_nodes(node vl, node v2) 

{ 

if (undirected) 

error_handler(1,"merge_nodes not implemented for undirected 
graphs."); 


for(int i=0; i<2; i++) 

{ 

if (vl->last_adj_edge[i] ) 

vl->last_adj_edge[i]->succ_adj_edge[i] = v2->first_adj_edge[i] ; 

else 

vl->first_adj_edge[i] = v2->first_adj_edge[i]; 
if (v2->first_adj___edge [i] ) 

{ v2->first_adj_edge[i]->pred_adj_edge[i] = vl->last_adj_edge[i]/ 

vl->last_adj_edge[i] = v2->last_adj_edge[i] ; 

} 

vl->adj_length [i] += v2->adj_length[i] ; 

v2->adj_length[i] = 0; 
v2->first_adj_edge[i] = 0; 
v2->last_adj_edge[i] = 0; 

} 

del_node(v2); 
return vl; 

} 


edge graph::add_edge(node v, node w, GenPtr inf) 

{ edge e; 

if (v->owner != this) 

error_handler(6, "new_edge(v, w) : v not in graph"); 
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if (w->owner != this) 

error_handler(6, "new_edge(v, w) : w not in graph"); 


if ( e_free.empty() ) 

{ e = (edge)std_memory.allocate_bytes(edge_bytes()); 
new (e) edge_struct(v, w, inf); 
e->id = ++max_e_index; 

} 

else 

{ e = (edge)e_free.pop 0 ; 
e->data[0] = inf; 
e->term[0] = v; 
e->term[l] = w; 
e-'>rev = nil; 

e“>succ_adj_edge[0] = nil; 
e->succ_adj_edge[1] = nil; 
e->pred_adj_edge[0] = nil; 
e->pred_adj_edge[1] = nil; 


e_list.append(e); 
graph_map* m; 

forall(m,map_list[1]) m->re_init_entry(e) ; 
return e; 


void graph::dealloc_edge(edge e) 

{ std_memory.deallocate_bytes(e,edge_bytes()); } 


void graph::del_adj_edge(edge e, node v, node w) 
{ if (undirected) 

{ v->del_adj_edge(e, 0,0) ; 
w->del_adj_edge(e,0,1) ; 

} 

else 

{ v->del_adj_edge(e,0,0); 
w->del_adj_edge(e, 1,1); 

} 

} 


void graph::ins_adj_edge(edge e, node v, edge el, node w, edge e2,int 
dl,int d2) 

{ 

// insert edge e 

// after(if dl=0)/before(if dl=l) el to adj_list of v 

// after(if d2=0)/before(if d2=l) e2 to in_list (adj_list) of w 

// (most general form of new__edge) 

if ( undirected ) 

{ if (v == w) 

error_handler(1,"new_edge(V,el,w, e2) : selfloop in undirected 
graph."); 

if (el && V != source(el) && v != target(el)) 
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error_handler(1,"new_edge(v, el, w, e2) : v is not adjacent to 

el."); 

if (e2 && w != source(e2) && w != target{e2)) 

error_handler(1,"new_edge(v,el, w, e2) : w is not adjacent to 

e2."); 

v->insert_adj_edge{e,el,0,0,dl); 
w->insert_adj_edge{e,e2,0,1, d2) ; 

} 

else 

{ if (el && V != source{el)) 

error_handler {1, "new__edge (v, el, w, e2) : v is not source of el.") 
if (e2 && w != source(e2) && w != target{e2)) 

error_handler(1,"new_edge(v,el, w,e2) : w is not target of e2.") 

v->insert_adj_edge(e,el,0,0,dl); 
w->insert_adj_edge(e,e2,1,1, d2) ; 

} 


edge graph::new_edge(node v, edge el, node w, edge e2, GenPtr i,int 
dl,int d2) 

{ 

// add edge (v,w,i) 

// after(if dl=0)/before(if dl=l) el to adj_list of v 

// after(if d2=0)/before(if d2=l) e2 to in_list (adj_list) of w 

// (most general form of new_edge) 

if ( undirected ) 

{ if (v == w) 

error_handler(1,"new_edge(v, el, w, e2) : selfloop in undirected 
graph."); 

if (el && V != source(el) && v != target(el)) 

error_handler (1, "new_edge (V, el, w, e2) : v .is not adjacent to 

el."); 

if (e2 && w 1= source(e2) && w != target(e2)) 

error_handler(1,"new_edge(v, el, w, e2) ; w is not adjacent to 

e2."); 

} 

else 

{ if (el && V != source(el)) 

error_handler(1,"new_edge(v, el, w, e2) : v is not source of el.") 
if (e2 && w != source(e2) && w != target(e2)) 

error_handler(1,"new_edge(v,el, w, e2) : w is not target of e2.") 

} 

pre_new_edge_handler(v,w); 
edge e = add_edge(v,w,i); 
ins_adj_edge(e,v,el,w,e2,dl, d2); 
post_new_edge_handler(e) ; 
return e ; 

} 

edge graph::new_edge(node v, edge el, node w, GenPtr i,int d) 

{ 

// add edge (v,w) after/before el-to adj_list of v 
// append it to in_list (adj_list) of w 

return new_edge(v,el,w,nil, i,d, 0); 
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} 


edge graph::new_edge(node v, node w, edge e2, GenPtr i,int d) 

{ 

// append edge (v,w) to adj_list of v 

// insert it after/before e2 to in_list (adj^list) of w 
return new_edge(v,nil,w,e2,i,d,0); 

} 

edge graph::new_edge(edge el, node w, GenPtr i, int dir) 

{ 

// add edge (source(el),w) after/before el to adj_list of source (el) 
// append it to in_list (adj_list) of w 

return new_edge(source(el),el,w,nil,i,dir,0); 

} 

edge graph::new_edge(node v, edge e2, GenPtr i, int dir) 

{ 

// append edge(v,target(e2)) to adj_list of v 

// insert it after/before e2 to in_list (adj_list) of target(e2) 
return new_edge(v,nil,target(e2),e2,i,0,dir); 

} 

edge graph::new_edge(edge el, edge e2, GenPtr i, int dirl, int dir2) 

{ 

//add edge (source(el),target(e2)) 

//after(dir=0)/before(dir=l) el to adj_list of source(el) 

//after(dir=l)/before(dir=l) e2 to in_list (adj_list) of target(e2) 

return new_edge(source(el),el,target(e2),e2,i,dirl,dir2); 

} 


edge graph::new_edge(node v, node w, GenPtr i) 

{ 

// append (v,w) it to adj_list of v and to in_list (adj_list) of w 
return new_edge(v,nil,w,nil,i,0,0); 

} 


node graph::split_edge(edge e, GenPtr node_inf, edges el, edges e2) 

{ 

// splits e into el and e2 by putting new node v on e 


//node V = source(e); 

node w = target(e); 

node u = add_node(node^inf); 

el = e; 

e2 = add_edge(u,w,e“>data[0]) / 
copy_edge_entry(e2->data[0]); 
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if (undirected) 

{ u->append_adj_edge(e2, 0,0) ; 
w->insert_adj_edge(e2,e, 0 ,1, 0 ) ; 
w->del_adj_edge(e,0, 1) ; 
e“>term[l] = u; 
u->append_adj_edge(e, 0, 1) ; 

} 

else 

{ u->append_adj_edge{e2,0,0); 
w->insert_adj_edge{e2, e, 1, 1, 0 ) ; 
w->del_adj_edge (e,1,1); 
e->term[l] = u; 
U“>append_adj_edge (e, 1, 1) ; 

} 

return u; 

) 


void graph::del_edge(edge e) 

{ node V = source(e); 
node w = target(e); 

if (v->owner != this) error_handler(10,”del_edge(e): e is not in G") 

pre_del_edge_handler(e); 

if (is_hidden(e)) restore_edge(e); 

if (e“>rev) e->rev->rev = nil; 

del_adj _edge(e,v,w); 

if (parent == 0) clear_edge_entry(e“>data[0]); 

e_list.remove(e); 
e_free.append(e); 

graph_map* m; 

forall(m,map_list[1]) 

{ int i = m->g_index; 

if (i > 0) m->clear_entry(e->data[i] ) ; 

} 

post_del_edge_handler(v, w) ; 

} 


void graph::hide_edge(edge e) 

{ 

if (is_hidden(e)) 

error_handler(1,"graph::hide_edge: edge is already hidden.”); 
pre_hide_edge_handler(e) ; 


node V = source(e); 
node w = target(e); 
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del_adj_edge(e,v,w); 

e_l ist.remove(e); 
h_list.append(e)/ 

e->id 1= 0x80000000/ 

post_hide_edge_handler(e) ; 

} 


void graph::restore_edge(edge e) 

{ 

if (!is_hidden(e)) 

error_handler (1,"graph::restore_edge: edge is not hidden."); 

pre_restore_edge_handler(e); 

node V = source (e); 
node w = target(e)/ 

h_list.remove(e); 
e_l ist.append(e); 

if (undirected) 

{ V“>append_adj_edge(e, 0, 0) ; 
w->append_adj_edge(e, 0, 1) ; 

} 

else 

{ V“>append_adj_edge(e, 0, 0) ; 
w->append_adj_edge(e, 1, 1) ; 

} 

e“>id = index(e); 
post_restore_edge_handler(e); 

} 


void graph::restore_all_edges() 

{ edge e = (edge)h_list.head() ; 
while (e) 

{ edge succ = (edge)h_list.succ(e) ; 
restore_edge(e); 
e = succ; 

} 

} 


void graph::move_edge(edge e,edge el,edge e2,int dl,int d2) 

{ if (is_hidden(e)) 

error_handler(1,"graph::move_edge: cannot move hidden edge."); 
node vO = source(e); 
node wO = target(e); 
node V = source(el); 
node w = target(e2); 
pre_move_edge_handler (e, V, w) ; 
del_adj_edge(e,source(e),target(e)); 
e“>term[0] = v; 
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e->term[l] = w; 

ins_adj_edge (e, v, el, w, e2, dl, d2) ; 
post_move_edge_handler(e,vO,wO); 

} 

void graph::move_edge(edge e,edge el,node w,int dir) 

{ if (is_hidden(e)) 

error_handler(1,"graph::move_edge: cannot move hidden edge.") 
node vO = source(e); 
node wO = target(e); 
node V = source(el); 
pre_move_edge_handler(e,v, w); 
del_adj_edge(e,source(e),target(e)); 
e->term[0] = v; 
e->term[l] = w; 

ins_adj_edge(e,source(el),el,w,nil,dir,0); 
post_move_edge_handler(e,vO,wO); 


void graph::move_edge(edge e, node v, node w) 

{ if (is_hidden(e)) 

error_handler(1,"graph::move_edge: cannot move hidden edge.") 
node vO = source (e); 
node wO = target(e); 
pre_move_edge_handler(e,v,w); 
del_adj_edge(e,source(e),target(e)) ; 
e->term[0] = v; 
e->term[l] = w; 

ins_adj_edge(e,v,nil,w,nil,0,0); 
post_move_edge_handler(e,vO,wO); 

} 


edge graph::rev_edge(edge e) 

{ if (is_hidden(e)) 

error_handler(1,"graph::move_edge: cannot move hidden edge.") 

node V = source(e); 

node w = target(e); 

pre_move_edge_handler(e,w, v); 

if (is_hidden(e)) // e hidden 

{ e->term[0] = w; 
e->term[l] = v; 
return e; 

} 

if (undirected) 

{ edge s = e->succ_adj_edge[0] ; 
edge p = e->pred_adj_edge[0] ; 
e->succ_adj_edge[0] = e->succ_adj_edge[1]; 
e->pred_adj_edge[0] = e->pred_adj_edge[1]; 
e->succ_adj_edge[1] = s; 
e->pred_adj_edge[1] = p; 
e->term[0] = w; 
e->term[l] = v; 

} 

else 

{ del_adj_edge(e,V,w); 
e->term[0] = w; 
e->term[l] = v; 
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ins_adj_edge(e,w,nil,v,nil, 0, 0) / 

} 

post_move__edge_handler (e, v, w) ; 
return e; 


void graph::rev_all_edges() 

{ if (!undirected) 

{ list<edge> L = all_edges(); 
edge e; 

forall(e,L) rev_edge(e); 


} 


} 


void graphdel_nodes(const list<node>& L) 

{ node v; 

forall(v,L) del_node(v) ; 

} 

void graph::del_edges(const list<edge>& L) 

{ edge e; 

forall(e,L) del_edge(e); 

} 

void graph::make_undirected() 

{ 

if (undirected) return; 

list<edge> loops; 
edge e; 

forall_edges(e,*this) 

if (source(e) == target(e)) loops.append(e) ; 
if ( 1 loops.empty() ) 

error_handler(0,"selfloops deleted in ugraph constructor"); 
fora11(e,loops) del_edge(e); 


/* adj_list(v) = out_list(v) + in_list(v) forall nodes v */ 
node v; 

forall_nodes(v,*this) 

{ 

// append in_list to adj_list 

if (v->first_adj_edge[1] == nil) continue; 


if (v->first_adj_edge[0] 
{ v->first_adj_edge[0] 
v->last_adj_edge[0] 
v->adj^length[0] 

} 


== nil) // move in_list to adj_list 
= v->first_adj__edge [ 1 ] ; 

= v->last_adj_edge[1] ; 

= v->adj_length[1]; 


else // both lists are non-empty 

{ v->last_adj_edge[0]->succ_adj_edge[0] = v->first_adj_edge[1] 
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v->first_adj_edge[1]->pred_adj_edge[1] = v->last_adj_edge[0] 
v->last_adj_edge[0] = v->last_adj_edge[1] ; 
v->adj_length[0] += v->adj_length[1] ; 

} 

v->first_adj_edge[1] = nil; 
v->last_adj_edge[1] = nil; 
v->adj_length[1] = 0; 


} 

undirected = true; 

} 


void graph: :raake_directed() 

{ 

if (!undirected) return; 

// for every node v delete entering edges from adj_list(v) 
// and put them back into in_list(v) 

node v; 

forall_nodes(v,*this) 

{ edge e = v->first_adj_edge[0]; 
while (e) 

if (v == target(e)) 

{ edge el = e~>succ_adj_edge[1] ; 
v->del_adj_edge(e, 0^1) ; 
v->append_adj_edge(e, 1,1) ; 
e = e 1 ; 

} 

else 

e = e->succ_adj_edge[0] ; 

} 

undirected = false; 


} 


void init_node_data(const graphs G,int i, GenPtr x) 
{ node v; 

forall_nodes(v^G) v->data[i] = x; 

} 


int graph::register_map(graph_map* M) 

{ int k = M->kind; 

M->g_loc = map_list[k].append(M); 

#if defined(LEDA_GRAPH_DATA) 
if (free_data[k] .empty() ) 
error_handler(1 , 

string("graph::register_map: all data (%d) slots 
used",data_sz[k])); 

#endif 
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return (free_data[k] .empty()) ? -1 : free_data[k] .pop (); 

} 


void graph::unregister_map(graph_map* M) 

{ int k = M->kind; 

map_list[k].del_item(M->g_loc) ; 

if (M->g_index > 0) free_data[k].push(M->g_index); 


node graph::choose_node() const 
{ int n = number_of_nodes() ; 
if (n == 0) return nil; 
int r = rand_int(0,n-1) ; 
node V = first_node() ; 
while (r--) v = succ_node(v); 
return v; 


edge graph::choose_edge() const 
{ int m = nuinber_of_edges () ; 
if (m == 0) return nil; 
int r = rand_int(0,m-1) ; 
edge e = first_edge(); 
while (r—) e = succ_edge(e); 
return e; 


face graph::choose_face() const 
{ int 1 = number_of_faces() ; 
if (1 == 0) return nil; 
int r = rand_int(0,1-1) ; 
face f = first_face() ; 
while (r--) f = succ_face(f); 
return f; 


// 


// old iterator stuff 
//- 


void graph::init_adj_iterator(node v) const 
{ adj_iterator->map_access(v) = nil; } 

bool graph::current_adj_edge(edge& e, node v) const 
{ return (e = (edge) adj__iterator->map_access (v) ) != nil;} 

bool graph:: next_adj_edge (edgeSt e, node v) const 
{ edge cur = (edge)adj_iterator->map_access(v) ; 
e = (cur) ? adj_succ(cur) : first_adj_edge(v); 
adj_iterator->map_access(v) = e; 
return (e) ? true : false; 
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} 


bool graph::next_adj_node(node& w, node v) const 
{ edge e; 

if (next_adj_edge(e, v) ) 

{ w = opposite(v,e); 
return true; 

} 

else return false; 

} 

bool graph::current_adj_node(node& w, node v) const 
{ edge e; 

if (current_adj_edge(e,v)) 

{ w = opposite(v,e); 
w = target(e); 
return true; 

} 

else return false; 


void graph::reset() const // reset all iterators 
{ adj_iterator->init(this,max_n_index+l, 0); 
node v; 

forall_nodes(V,*this) adj_iterator~>map_access(v) = nil; 
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+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH^ Postfach 151101, 66041 Saarbruecken, FRG 

+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

'k'k’k^'k^'k'k'k-^'k'k'k'k-^-^'k'k-^'k'k-k'k’k'k'k’k'k'k'k'k'k-^’k'k'k'k’k'k'k'k'k'k'k'fe'k'k'k'k'k'k'k'k'k'fe^'k'k'k'k'k'k'k'k'k’k’k'k'k’k'k-k 

#include <LEDA/planar_map.h> 


void planar_map;:init_face_entries() const 
{ face f; 

forall_faces(f,*this) init_face_entry(f->data[0]); 

} 

void planar_map::copy_face_entries() const 
{ face f; 

forall_faces(f,*this) copy_face_entry(f->data[0 ] ); 

} 

void planar_map::clear_face_entries() const 
{ face f; 

forall_faces(f,*this) clear_face_entry(f->data[0] ); 

} 


node planar_map::new_node(const list<edge>& el) 

{ 

if (el.length() < 2) 

error_handler(1,”planar_map;:new_node(el,i): el.length() < 2.”); 

list_item it = el.first(); 

edge eO = el [it]; 

it = el.succ(it); 

face f = adj_face(eO); 

edge e; 
forall(e, el) 

{ if (adj_face(e) != f) 

error___handler (1, **planar_map: :new_node: edges bound different 
faces . **) ; 

} 

e = el[it] ; 
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it = el.succ(it); 

GenPtr face_inf = f->data[0]/ 
copy_face_entry(face_inf) / 

edge x = new_edge(eO,e); 
face fx = adj_face{reversal{x))/ 
clear_face_entry{fx->data[0] ) ; 
fx->data[0] = face_inf; 

edge el = split_edge(x); 

while(it) 

{ copy_face_entry(face_inf); 
el = new_edge(el^el[it]); 
face fx = adj_face(reversal(el))/ 
clear_face_entry{fx->data[0] ) ; 
fx->data[0] = face_inf/ 
it = el.succ(it); 

} 

return source(el); 


131 


+ 

+ LEDA 3.5.1 
+ 

+ _pq_tree.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

*******/ 

// - 

// PQ_TREES 

// 

// R. Hesse, E. Kalliwoda, D. Ambras (1996/97) 

// 

// - 


#include <LEDA/pq_tree.h> 
#include <LEDA/queue.h> 

const int 

D_NODE_INVALID=0, 
D_NODE_PNODE =1, 
D_NODE_QNODE =2, 
D_NODE_LEAF =3, 

D_NODE_DIR =4, 

D_NODE_UNMARKE D = 6, 
D_NODE_UNBLOCKED =7, 
D_NODE_BLOCKED =8, 
D_NODE_QUEUED =9, 

D_NODE_EMPTY =10, 

D_NODE_FULL =11, 

D_NODE_PARTIAL =12, 

D NODE DOUBLE PARTIAL =13; 


pq_node_struct::pq_node_struct() 


leaf_index = 
child_count = 

parent 

link_one_side 
link_other_side 
right_most 
left_most 

type 


0 ; 

0 ; 

= NULL; 

= NULL; 

= NULL; 

= NULL; 

= NULL; 

D NODE PNODE; 
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parent_type = D_NODE_QNODE; 
node reset(); 


void pq_node_struct::node_reset() 

{ 

mark = D_NODE_UNMARKED; 

status = D_NODE_EMPTY; // vielleicht auf FULL umstellen *** 

pert_leaf_count = 0; 
pert_child_count = 0; 
full_child_count = 0; 

part__childl = NULL; 
part_child2 = NULL; 


pq_tree:;pq_tree(int 

{ 

successful 

too_many_part 

leaves_size 

blocked_chain_count 

blocked_nodes_count 

root 

pseudo_root 

pseudo_root->leaf_i 


ize) 

= true; 

= false; 

= Isize; 

= 0 ; 

= 0 ; 

= NULL; 

= new pq_node_struct; 
[ex = 2; 


if (Isize > 0) leaves_init(lsize+1); 
else leaves = NULL; 

} 


// - 

// TEMPLATES 
//- 


void pq_tree : ; f ix_part_child_direction {pq_node x, bool part_child2_too) 

{ 

pq_node k; 

pq_node childl= x->part_childl; 

pq_node child2= x->part_child2; 

if {childl->left_most->status == D_NODE_FULL) 

{ k= childl->left_most; 

childl->left_most = childl->right_most; 
childl->right_most = k; 


if (part_child2_too) 

if {child2->right_most->status == D_NODE_FULL) 
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{ k= child2->left_niost ; 

child2->left_most = child2->right_most; 
child2->right_niost = k; 

} 

) 


inline void pq_tree::append_as_right_child(pq_node parent, pq_node 
child) 

{ 

pq_node 1= parent->right_most; 

if (l->link_one_side) 

l->link_other_side = child; 

else 

l->link_one_side = child; 

child->link_one_side =1; 
child->link_other_side= NULL; 

parent->right_most = child; 

} 


pq_node pq_tree::hang_down_full_children(pq_node x, pq_node z) 

{ 

int x_full_child_count= x->full_child_count; 

pq__node y; 


if (x_full_child_count == 0) return NULL; 

if (x_full_child_count === 1) 

{ 

y= remove_from_siblings(x, x->right_most); 
y“>parent = z; 
y->parent_type = z->type; 
append_as_right_child(z, y); 

z~>child_count++; 

z->full_child_count++; 

z->pert_leaf_count = x->pert_leaf_count; 
x->full_child_count = 0; 
x->child_count—; 

return y; 

} 


pq_node 1; 

y = new pq_node_struct; 
y->parent = z; 
y->parent__type = z->type; 
y->status = D_NODE_FULL; 
processed.push(y); 
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y->left_most = x->right_most; 
y->right_most= x->right_most; 

y->left__most->parent = y; 

1 = NULL; 

for(int i = 1; i < x_full_child_count; i++) 
{ go_to_sibling(y->left_most, 1); 
y->lef t_most-'>parent = y; 

} 


pq_node k, sibling; 
pq_node y_left= y->left_most; 

if (y_lef t->link_one_side == 1) 

{ k= y_lef t->link_other__side; 

y_left->link_other_side = NULL; 

} 

else 

{ k= y_left->link_one_side; 

y_left->link_one_side = NULL; 

} 

if (x != z) 

{ x->right__most = k; 

append_as_right_child(z, y) ; 
sibling= NULL; 

} 

else 

{ x->right_most = y; 
y->link_one_side = k; 
sibling= y; 

} 

if {k->link_one_side == y_left) 
k->link__one_side = sibling; 
else 

k->link_other_side = sibling; 


y->child_count = y->full_child_count = x->full_child_count 

x->full_child_count = 0; 
x->child_count -= y->child_count ; 

y->pert__leaf_count = x->pert_leaf_count; 
if (x->part_childl) 

y->pert_leaf_count x->part_childl->pert_leaf_count; 
if (x->part_child2) 

y->pert_leaf_count -= x->part_child2->pert_leaf_count; 

z->child_count++; 

z->full_child_count++; 

z->pert_leaf__count += y->pert_leaf_count; 
return y; 

} 
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pq_node pq_tree::remove_from_siblings(pq_node parent, pq_node child) 

{ 

if (child->link_one_side) 

if (child->link_one_side”>link_one_side == child) 

child->link_one_side->link_one_side = child->link_other_side 

else 

child->link_one_side->link_other_side= child“>link_other_side 

if {child->link_other_side) 

if (child->link_other_side“>link_one_side == child) 

child->link_other_side->link__one_side = child->link_one_side 

else 

child->link other side“>link other side= child->link one side 


if (parent->right_most == child) 

parent->right_most = child->link_one_side ? 

child“>link_one_side : child->link_other_side; 

if (parent“>left_most == child) 

parent->left_most = child”>link_one_side ? 

child->link one side : child”>link other side; 


if {parent->part_childl == child) 

{ parent->part_childl= parent->part_child2; 
parent“>part_child2= NULL; 

) 

if (parent->part_child2 == child) parent->part_child2= NULL; 
return child; 

} 


void pq_tree : : replace_in__siblings (pq_node x, pq_node z) 

{ 

Z”>parent = x->parent; 

Z”>parent_type = x~>parent_type; 

z->link_one__side = x->link_one_side; 
z->link_other_side = x->link_other_side; 

if (Z“>link_one_side) 

if (z->link_one_side->link_one_side == x) 
Z“>link_one_side->link_one_side = z; 

else 

Z“>link__one_side”>link_other_side = z; 

if {z->link_other_side) 

if {z->link_other_side->link_one_side == x) 
z->link_other_side“>link_one_side = z; 

else 

z->link_other_side“>link_other_side = z; 
if (z->parent) 

{ if (z->parent->right_most == x) z->parent“>right_most = z; 
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if (z->parent->left_most == x) z->parent->left_most = z; 

} 

} 


bool pq_tree : : set_as_part_child (pq_nocie x) 

{ 

x->status = D_NODE_PARTIAL; 

if (x->parent->part_childl) 

{ if (x->parent->part_child2) 

{ too_many__part = true; 
return false; 

} 

else x->parent->part_child2 = x; 

} 

else x->parent->part_childl = x; 
return true; 

} 


inline void pq_tree::append_as_left_child(pq_node parent, pq_node child) 

{ 

pq_node 1= parent->left_most; 
child->parent = parent; 

child->parent__type = D_NODE_QNODE; // ist immer so 

if (l->link_one_side) 

l->link_other_side = child; 

else 

l->link_one_side = child; 

child->link_one_side = NULL; 
child->link_other_side= 1; 

parent->left_most = child; 

} 


void pq_tree::delete_part_node_parent(pq_node x) 

{ 

replace_in_siblings(x, x->part_childl); 

if (x == root) 

{ root= x->part_childl; 
root->parent = NULL; 
root->link_one_side = NULL; 
root->link_other_side = NULL; 

} 

delete x; 

} 
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bool pq_tree : : template_PQLl (pq_node x, bool is__pseudo_root) 

{ 

#ifdef _DEBUG_PQ_TREE 

if (x->type == D_NODE_PNODE) cout « "PI"; 
else 

if (x->type == D_NODE_QNODE) cout « "Ql"; 
else cout << "LI"; 

#endif 


if (X“>type == D_NODE_QNODE) 

{ pq_node k; 

pq_node last_k= NULL; 

if (x->part_childl) return false; 
if (x == pseudo_root) return true; 

k = x->right_most; 
while (k) 

{ if (k“>type != D_NODE_DIR && k->status == D_NODE_EMPTY) 
return false; 
go_to_sibling(k, last_k); 

} 

} 

else 

if (x->full child count != x->child count) return false; 


if (x->parent_type == D_NODE_PNODE && x->parent && 
x->parent->right_most != x ) 

{ 

remove_from_siblings(x->parent, x) ; 
append_as_right_child(x->parent, x) ; 

} 

x->status = D_NODE_FULL; 
processed.push(x); 

if (is_pseudo_root) 

{ 

pseudo_root->left_most = x; 
pseudo_root->right_inost = x; 

} 

else 

if (x->parent) x->parent->full_child_count++; 
return true; 


bool pq_tree::tempiate_P3(pq_node x, bool is_pseudo_root) 

{ 

pq_node y, z; 
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#ifdef _DEBUG_PQ_TREE 

if (is_pseudo_root) cout « "P2"; 

else cout << ”P3”; 
assert(x != root); 
cons_pq_tree(root,"b","P2/3") ; 
#endif 

if (x->part_childl) return false; 

if (x->full_child_count > 1) 

y= hang_down_full_children(x, x); 
else 

y= x->right_most; 

if (is_pseudo_root) 

{ 

pseudo_root->left_most = y; 
pseudo_root->right_most = y; 
x->node_reset(); 

return true; 

} 


if (x->child_count > 2) 

{ 

z = new pq_node_struct; 
z->mark = D_NODE_UNBLOCKED; 

//for correct blocked chain handling in reduce() 

if (x == pseudo_root->left_most) pseudo_root->left_most = z; 
if (x == pseudo_root->right_most) pseudo_root->right_most = z; 

replace_in_siblings(x, z) ; 

x->right_most = y->link_one_side ? 

y->link_one_side : y->link_other_side; 

if (x->right_most->link_one_side == y) 
x->right_most->link_one_side = NULL; 
else 

x->right_most->link_other_side = NULL; 

x->child_count—; 
x->node reset(); 


x->parent = y->parent = z; 
z->left_most = x; 
z->right_most = y; 

x->link_one_side = NULL; 
x->link_other_side= y; 

y->link_one_side = NULL; 
y->link_other_side= x; 
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z->pert_leaf__count = x->pert_leaf_count ; 
z~>full_child_count = 1; 
z->child_count = 2; 

} 

else z = x; 

z->type = D_NODE_QNODE; 

z~>left_most->parent_type = D_NODE_QNODE; 
z->right_most“>parent_type= D_NODE_QNODE; 

return set_as_part_child(z); 

} 


bool pq_tree::template_P5(pq_node x, bool is_pseudo_root) 

{ 

pq_node y, k, z; 

#ifdef _DEBUG_PQ_TREE 

if (is_pseudo_root) cout << "P4''; 

else cout << ”P5”; 
assert(x != root); 
cons_pq_tree(root,"b”,"P4/5"); 

#endif 


if (x->part_child2 || too_many_part ) return false; 

// if (!x->part childl || x->part child2 || too many part ) 


// return false; 

// "II too_many_part" muss bleiben, 

fix_part_child_direction(x); 

z = X“>part_childl; 

y= hang_down_full_children(x, z) ; 
gibt ? *** 

if (is_pseudo_root) 

{ 


x->part_childl darf nicht !? 

// Teste, ob es voile Kinder 

// ( sonst y= NULL ) 

// it's P4 


pseudo_root->right_most = y; 

// suche letztes voiles Kind: 

k = y->link_one_side ? 

y->link_one_side : y->link_other_side; 

while (k->status == D_NODE_FULL || k->type == D_NODE_DIR) 
go_to_sibling(k, y); 

pseudo_root->left_most = y; // y= letztes voiles Kind 

z->node_reset(); 
if (x“>child count == 1) 
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delete_part_node_parent(x); 
else 

x->node_reset(); 
return true; 


if (x->child_count > 2) 

{ 

z->pert_leaf_count = x->pert_leaf_count; 
x->child_count—; 
x->pert_leaf_count = 0; 

x->part_childl = NULL; 
remove_from^siblings{x, z); 
replace_in_siblings(x, z); 
set_as_part_child(z); 
append_as_left_child{z, x); 

x->node_reset(); 

} 


else 

{ 

if (X“>child_count == 2) 

{ 

y = (x“>right_most == z) ? 

x->left_most : x->right_most; 
append_as_left_child{z, y) ; 

} 


// for correct blocked chain handling in reduce(): 
z->mark = D_NODE_UNBLOCKED; 

if (x == pseudo_root->left_most) pseudo_root->left_most 
if (x == pseudo_root->right_most) pseudo_root“>right_most 

replace_in_siblings(x, z); 
set_as_part_child(z); 

delete x; 

) 

return !too_many_part; 


bool pq_tree::template_P6{pq_node x) 

{ 


#ifdef _DEBUG_PQ_TREE 
cout << "P6"; 

#endif 
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pq_node k, 1/ 

if (!x->part_child2 || too_many_part) return false; 

fix_part_child_direction(x, true); 
hang_down_full_children (x, x*->part_childl) / 


1 = x*->part_childl->right_most; 
k = l“>link_one_side ? 

l->link_one_side : l->link_other_side; 

while (k->status == D_NODE_FULL | | k-'>type == D_NODE_DIR) 
go_to_sibling(k, 1); 

pseudo_root->left_most = 1; // linkestes voiles Kind von x- 

>partl 

l->parent = x->part_childl; 


1 = x*->part_child2->lef t_most; 
k = l->link_one_side ? 

l->link_one_side : l->link_other_side; 

while (k->status == D_NODE_FULL || k->type == D_NODE_DIR) 
go_to_sibling(k, 1); 

pseudo_root->right_most = 1; // rechtestes voiles Kind von x- 

>part2 


k= x->part_childl/ 

1= reinove_f rom_siblings (x, x->part_child2 ) ; 
x->child_count—; 

if {k->right_most->link_one_side) 

k->right_most’->link_other_side = l->left_inost ; 
else 

k->right_most->link_one_side = l->left_most ; 

if (l-’>left_most-’>link_one_side) 

l*->left_most'->link_ot he reside = k->right_most / 
else 

l->left_most->link_one_side = k->right_most ; 

k->right_most = l->right_most; 
k*->right_most-’>parent= k; 


delete 1; 

x->part_childl-’>node_reset () ; 

if (x->child_count == 1) 

delete_part_node_parent(x) ; 
else 

x*->node_reset () ; 
return true; 
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bool pq_tree::template_Q2(pq_node x) 

{ // template for a Q-node with empties and/or 1 partial 

child 


#ifdef _DEBUG_PQ_TREE 
cout << "Q2" << endl; 

if (pseudo_root->leaf_index > 1) show("in Q2”,x)/ 

//show(”in Q2",x); 

//show(”in Q2",root); 

#endif 

if (x->part_child2) return false; 

// if (!x->part_childl || x->part_child2 || too_many_part) // darf 

nicht ! 

// return false; 


pq node 

k; 

// 

pq node 

1; 

// 

pq node 

11,12; 

// 

pq node 

m, n; 

// 

pq node 

pl=NULL; 

// 

pq node 

dl; 

// 

^ • • • 

pq node 

d2; 

// 

pq_node 

aux leftni=NULL; 


pq_node 

aux rightni=NULL; 

// 

bool 

full_found=false; 

// 

bool 

a,b,c,d; 

// 


to run through the children 

dito 

dito 

dito, but see text 
the part_child 

the side the full*s will be turned 
and the other side 

dummies for pseudo_root->end_most ’ s 

turn the full's to the outside 
for testing of blocked chain 


if (pi = x->part_childl) 

{ 

11 = 12 = X“>part_childl; 

m = pl->link_one_side; 
skip_dir(m,11); 

n = pl->link_other_side; 
skip_dir(n,12); 

if (x->full_child_count) 

{ 

a = m ? m->status == D_NODE_FULL ; 0; 
b = n ? n->status == D_NODE_FULL : 0; 

if (!(a ^ b)) return false; // xor; (both (full)) or (both (NULL 
or empty)) 

if (x->full_child_count == 1) 

{ 
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if (a) 

{ dl = pl->link_one_side; 
d2 = pl->link_other_side; 
k = m; 

1 = 11 ; 

} 

else 

{ d2 = pl->link_one_side; 
dl = pl->link_other_side; 
k = n; 

1 = 12 / 

} 

go_to_sibling (k, 1) ; 

skip_dir(k, 1); 
aux_leftm = 1; 

// in case of a blocked chain the end_most*s of pseudo_root are already 
used 

// and valid 

// next reduce round Q2 will match pseudo_root as the father of the 
blocked chain 

full_found = true; 

} 

else 

{ 

if (a) 

{ k = m; 1 = 11; } 

else 

{ k = n; 1 = 12; } 

for(int i = 2; i <= x~*>full_child_count; i+-f) 

{ go_to_sibling(k, 1) ; 

skip_dir(k, 1) ; 

if ( !k M k-“>status ! =»= D_NODE_FULL ) return false; 

//there is an empty between the full’s 
//or not all full’s are at one side 

} 

full_found = true; 
go_to_sibling(k, 1) ; 

skip_dir(k, 1) ; 
aux_leftm =1; 

if (a) 

{ dl = pl~*>link_one_side; d2 = pl->link_other_side; } 
else 

{ d2 = pl~>link_one_side; dl = pl->link_other_side; } 

} 

//if (x->full_child_count) 

//... no full’s, only a part_child 
if (pseudo_root*“>status == D_NODE_FULL) 


} 

else 

{ 
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{ // ROOT(T,S) is reached 

dl = pl->link_one_side; 
d2 = pl->link_other_side; 

if (dl->type == D_NODE_DIR) 

{ 1 = pi; 
k = dl; 

skip_dir(k,1); 
aux_leftm = 1; 

} 

else 

if (pl->right_most->status == D_NODE_FULL) 
aux_leftin = pl->right_most ; 
else 

aux_leftm = pl->left_most/ 

} 

else 

{ if (m && n) return false; //-part_child is between 

empties 

dl 

will be 

d2 

end_most 

} 

} 

} //end of "if (pi = x->part_childl)" 
else 

{ // there's no part_child 

if (x->left_most->status == D_NODE_FULL) 

1 = x->left_most; 
else 

{ 1 = x->right_most/ 

if ( l~>status != D_NODE_FULL) return false; 

} 

// if ( pseudo_root->type != D_NODE__QNODE | | x == pseudo_root) 

// aux_rightm = aux_leftm =1; 

aux_rightm = aux_leftm = 1; 

if (x->full_child_count > 1) 

{ 

k = l->link_one_side ? l->link_one_side : l“>link_other_side; 
skip_dir(k,1); 

if (k->status == D_NODE_FULL) 

{ for(int i = 2; i <= x->full_child_count; i++) 

{ skip_dir(k,1); 

if ( k->status != D_NODE_FULL) return false; 
go_to_sibling(k, 1); 

} 

// if ( pseudo_root->type != D_NODE_QNODE || x == pseudo_root ) 

// if ( pseudo_root->status == D_NODE_FULL) 

// { 

skip_dir (k, 1) ; 
aux_leftm = 1; 

// } 


= NULL; //-part_child*s full endmost 

= m ? pl->link_one_side : pl->link_other_side; 

//turned outside and becomes an 
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} 

else return false; // empties are intermingled with full's 

} 

} //else from "if (pi = x->part_childl)" 

//no empty's and no part_child between full's 
//template applicable, now the replacement: 

if (pl) 

{ 

if (pl->right_most->status == D_NODE_FULL) 

{ m = pl->right_most; n = pl“>left_most; } 
else 

{ m = pl“>left_most, n = pl“>right_most; } 

if (m->link_one_side) 

m->link_other_side = dl; 
else 

m~>link_one_side = dl; 
if (dl) 

if (dl->link_one_side == pl) 
dl->link_one_side = m; 
else 

dl“>link_other_side = m; 

if (n->link_one_side) 

n->link_other_side = d2; 
else 

n->link_one_side = d2; 
if (d2) 

if (d2->link_one_side == pl) 
d2->link_one_side. = n; 
else 

d2“>link_other_side = n; 

if (pl == x->left_most) 

{ x->left_most = full_found ? n : m; 
x->left_most“>parent = x; 

} 

if (pl == X“>right_most ) 

{ x->right_most = full_found ? n : m; 

X“>right_most“>parent = x; 

} 


X“>part_childl = NULL; 

//reversed (if necessary) and chained 

X“>full_child_count += pl“>full_child_count; 
delete pl; 

1 = dl; 
k = m; 

while (k && k“>status == D_NODE_FULL) 

{ k->parent = x; 

go_to_sibling(k, 1); 
skip_dir(k,1); 

} 
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aux__rightm = 1; 


} 

if (aux_leftm && aux_rightm) 

{ c = aux_leftm“>link_one_side && aux_leftin->link_other_side; 
d = aux_rightni->link_one_side && aux_rightm->link_other_side; 
if (c && d) x->status = D_NODE_DOUBLE_PARTIAL; 

} 


if (pseudo_root->status == D_NODE_FULL |1 x == pseudo_root){ 
pseudo_root->left_inost = aux_leftm; 
pseudo_root->right_most = aux_rightm; 

} 

else 

if (x->status == D NODE_DOUBLE PARTIAL) return false; 


if (x != pseudo_root) 

{ // father pointer of x is valid, that means != NULL 
if (pseudo_root->status == D_NODE_FULL) // ROOT(T,S) reached 
x->node_reset(); 

else 

{ X“>status = D_NODE_PARTIAL; 
if (x->parent->part_childl) 

{ if (x->parent->part_child2) 

{ too_many_part = true; 
return false; 

} 

else 

x->parent“>part_child2 = x; 

} 

else 

X“>parent->part_childl = x; 

//x becomes one of the X“>parent->part_children 

} 

} 

return true; 


bool pq_tree::tempiate_Q3(pq_node x) 

{ // template for a Q-node with exactly 2 partial 

children 


#ifdef _DEBUG_PQ_TREE 
cout « "Q3”; 

#endif 

if ( pseudo_root->status != D_NODE_FULL I1 
too_many_part I I 
!x->part_child2 ) return false; 
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pq_node 

k; 

pq node 

1; 

pq node 

pl; 

pq_node 

dl; 

pq node 

P2; 

pq node 

d2; 

pq node 

mm, m; 

pq node 

nn, n ; 

pq_node 

11,12; 


//to run through the children 
//dito 


//a part_child ... 

//... and its neighbour in direction to the full*s 

//dito 

//dito 


//to run through the children 

//dito 

//dito 


unsigned char cc; //for a check 


bool a,b; 


11 = 12 = pi = x->part_childl; 
m = pl“>link_one_side; 
skip_dir(m,11); 

n = pl->link_other_side; 
skip_dir(n,12); 

11 = 12 = p2 = x->part_child2; 
mm = p2->link_one_side; 
skip_dir(mm,11); 
nn = p2->link_other_side; 
skip_dir(nn,12) / 

if (X“>full_child_count) 

{ // X has full children 

cc = m ? (m->status == D_NODE_PARTIAL ? 1 : 

(m->status == D_NODE_FULL ? 2 : 0)) : 0; 

cc += n ? (n->status == D_NODE_PARTIAL ? 1 : 

(n->status == D_NODE_FULL ? 2 : 0)) : 0; 

if (cc != 2) return false; // OH GOTT !! *** 

dl = (m && m“>status == D_NODE_FULL) ? pl->link_one_side : pi- 
>link_other_side; 

cc = mm ? (mm->status == D_NODE_PARTIAL ? 1 : (mm->status == 
D_NODE_FULL ? 2 : 0)) : 0; 

cc += nn ? (nn->status == D_NODE_PARTIAL ? 1 : (nn->status == 
D_NODE_FULL ? 2 : 0)) : 0 ; 

if (cc != 2) return false; // *** 

d2 = (mm && mm->status == D_NODE_FULL) ? p2->link_one_side : p2- 
>link_other_side; 

/★ 

explanation: 
check values for 
NULL empty part full 
0 0 12 

One sibling is "NULL*' or "empty" and the other is "full" is a 
necessary 

condition here for a valid Q3 situation (the rest of the test 
follows). 
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*/ 

if (x->full_child_count > 1) 

{ 

if (mm && mm->status == D_NODE_FULL) 

{ d2 = p2“>link_one_side; k = mm; 1 = 11;} 
else 

{ d2 = p2->link_other_side; k = nn; 1 = 12;} 

for(int i = 2; i <= X“>full_child_count; i++) 

{ go_to_sibling(k, 1); 

skip_dir(k, 1) ; 

if (!k II k->status != D_NODE_FULL) return false; 

// because there is an empty or an part_child between the 

full’s 

} 


go_to_sibling(k, 1); 

skip_dir(k,1); 

if (!k II k->status != D_NODE__PARTIAL) return false; 

} 

} 

else 

{ // X has no full child 

a = m ? m->status == D_NODE_PARTIAL : 0; 
b = n ? n->status == D_NODE_PARTIAL : 0; 
dl = a ? pl“>link_one_side : pl“>link_other_side; 
if (!(a b)) return false; 

a = mm ? mm“>status == D_NODE_PARTIAL : 0; 
b = nn ? nn“>status == D_NODE_PARTIAL : 0; 
d2 = a ? p2“>link_one_side : p2->link__other_side; 
if (!(a ^ b)) return false; 

} 

//no empties and no part_child between full's 
//template is applicable, now the replacement: 

pq_node ml,m2; //the full end_most of a part_child 
pq_node nl,n2; //the empty end_most of a part_child 
pq_node o; //a dummy 

if (dl == p2) 

{ // the partial children are neighbours 

if (pl->right_most“>status == D_NODE_FULL) 

{ ml = pl“>right_most; 
nl = pl->left_most; 

} 

else 

{ ml = pl->left_most; 
nl = pl-->right_most ; 

} 

if (p2->right_most->status == D_NODE_FULL) 

{ m2 = p2->right_most; 
n2 = p2“>left_most; 

} 
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else 

{ m2 = p2->left_most; 
n2 = p2->right_most; 

} 


o = 

= (pl->link 

one 

side 

== dl) ? 

pl~>link other side : 

pl- 

>link_ 

_one_side; 






if 

(ml->link 

one 

side) 

ml->link 

other side = m2; else 

ml~ 

>link 

one side = 

m2; 





if 

(m2->link 

one 

side) 

m2->link 

_other_side = ml; else 

m2- 

>link 

one side = 

ml ; 





if 

(nl~>link_ 

one_ 

side) 

nl->link_ 

other side = o; else 

nl- 

>link 

one side = 

o; 






if (o) 

if (o~>link:_one_side == pi) 
o~>link_one_side = nl; 
else 

0 “>link_other_side = nl; 

o = (p2->link_one_side == d2) ? p2->link_other_side : p2~ 

>link_one_side; 

if (n2~>link_one_side) 

n2~>link_other_side = o; 
else 

n2->link_one_side = o; 
if (o) 

if (o->link_one_side == p2) 
o~>link_one_side = n2; 
else 

o~>link_other_side = n2; 

dl = m2; 
d2 = ml; 

//reversed and chained 

} 

else 

{ // full children between the partial 

if (pl->right_most->status == D_NODE_FULL) 

{ ml = pl~>right_most; 
nl = pl->left_most; 

} 

else 

{ ml = pl~>left_most; 
nl = pl->right_most ; 

} 

o = (pl~>link_one_side == dl) ? pl->link_other_side : pl~ 
>link_one__side; 

if (ml“>link_one_side) ml->link_other_side = dl; else ml 

>link__one__side = dl; 

if (dl~>link_one_side == pi) dl->link_one_side = ml; else dl- 
>link_other_side = ml; 

if (nl->link_one_side) nl->link_other_side = o; else nl 

>link one side = o; 
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if (o) 

if (o->link_one_side == pi) 
o->link_one_side = nl; 
else 

o->link_other_side = nl; 

if (p2->right_most->status == D_NODE_FULL) 

{ m2 = p2->right_most; 
n2 = p2->left_most; 

} 

else 

{ m2 = p2->left_most; 
n2 = p2->right_most ; 

} 

o = (p2->link_one_side == d2) ? p2->link_other_side : p2- 
>link_one_side; 

if (m2->link_one_side) m2->link_other_side = d2; else m2- 

>link_one_side = d2; 

if (d2->link_one_side == p2) d2->link_one_side = m2; else d2- 
>link_other_side = m2; 

if (n2->link_one_side) n2“>link_other_side = o; else n2- 

>link one side = o; 


if (o) 

if (o->link_one_side == p2) 
o->link_one_side = n2; 
else 

o->link_other_side = n2; 

// reversed and chained 

} 

1 = dl; 
k = ml; 

while (k->status == D_NODE_FULL) 

{ go_to_sibling(k, 1); 
skip_dir(k,1); 

} 

pseudo_root->left_most = 1; 

1 = d2; 
k = m2; 

while (k->status == D_NODE_FULL) 

{ go_to_sibling(k, 1) ; 
skip_dir(k,1); 

} 

pseudo_root->right_most =1; //end_most of pseudo_root now 

valid 

if (pi == x->left_most) 

{ x->left_most = nl; 
nl->parent = x; 

} 

else 


is 


151 


if (p2 == x-->lef t_most) 

{ x->left_most = n2; 
n2->parent = x; 

} 

if (pi == x~>right_most) 

{ x~>right_most = nl; 
nl~>parent = x; 

} 

else 

if (p2 == x->right_Tnost ) 

{ x->right_most = n2; 
n2~>parent = x; 

} 

if (x != pseudo_root) x~>node_reset() ; 

pseudo_root->lef t_Tnost->parent = x; 

delete pi; 
delete p2; 

x~>part_childl = x~>part_child2 = NULL; 
return true; 


pq_tree;:~pq_tree() 

{ 

if (!successful) del_subtree(root) ; 
delete pseudo_root; 
delete[] leaves; 


bool pq_tree::reduction(list<int>& S) 

{ 

list_item lit= processed.first() ; 
if (lit) 

{ do processed.inf(lit)->node_reset() ; 
while( lit= processed.succ(lit) ); 

processed.clear (); 

} 

pseudo_root->type = D_NODE_PNODE; 
pseudo_root“>node_reset() ; 


successful = bubble(S) && reduce(S); 
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#ifdef _DEBUG_PQ_TREE 

if (!successful) show{”not successful”, root) ; 
#endif 

return successful; 


inline void pq_tree::frontier{list<int> &F) 
{ F.clear {); 

sequence(F, root); 

} 


void pq_tree::sequence{list<int>& S, pq_node x, pq_node 1) 

{ 

if {x->type == D_NODE_LEAF) 

{ S.append{x->leaf_index) ; 
return; 

} 


pq_node k = x->left_most; 
pq_node r = x->right_most; 

do 

{ if {k->type == D_NODE_DIR) 
{ int i = S.popO +1; 
sequence S 


// insert one more DIR-ptr in the 
// with respect to its direction 


if (k->link_one_side == 1) 

S.push{k->leaf_index); 
else 

S.push{“k“>leaf_index); ‘ 

S.push (i); 

processed.push {k); // we have to delete the DIR-ptr in 

update{) 

} 

else 

sequence(S,k); 
go_to_sibling{k, 1); 

} while {1 != r); 

} 


bool pq_tree::bubble{list<int>& S) 

{ 

queue<pq_node> Q; 

pq_node x, y, z, k, 1; 

int m, blocked_found; 

#ifdef DEBUG PQ TREE 
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cout << endl; 

#endif 

root_reached = 0/ 
blocked_chain_count = 0; 
blocked_nodes_count = 0; 

pseudo_root->left_most = NULL; 
pseudo_root“>right_most= NULL; 

forall(m, S) 

{ X = leaves[m]; 

x->mark = D_NODE_QUEUED; Q.append(x); 

} 

#ifdef _DEBUG_PQ_TREE 

printf(” %d ”,thiS“>pseudo_root->leaf_index); 
cons_pq_tree(root,”a”,"bubble”); 

// show("in bubble",root) ; 

#endif 


while ((Q.sizeO + blocked_chain_count + root_reached) > 1) 

{ 

if (Q,empty()) return false; 

X = Q.pop(); 

if ( x->parent_type == D_NODE_PNODE 1| !x->link_one_side || ix- 

>link_other_side) 

x->mark = D_NODE_UNBLOCKED; 
else 

{ // try to make it valid in constant time 
x->mark = D_NODE_BLOCKED; 
k = x->link_one_side; 1 = x; 
skip_dir (k, 1) ; 

if ( k~>mark == D_NODE_UNBLOCKED) 

{ //the link_one_side“Sibling has a valid parent 
x->mark = D_NODE_UNBLOCKED; 
x->parent = k“>parent; 

} 

else 

{ k = x->link_other_side; 1 = x; 
skip_dir(k,1); 

if ( k“>mark == D_NODE_UNBLOCKED) 

{ //the link_other_side-sibling has a valid parent 
x->mark = D_NODE_UNBLOCKED; 
x~>parent = k->parent; 

} 

} 


if (X“>mark == D_NODE_UNBLOCKED) 

{ //x has got a valid parent 
y = x-->parent; 
z = x->link_one_side; 

if ( z && (Z“>mark == D_N0DE_BLOCKED 11 z->type == D_NODE_DIR) ) 

{ 1 = x; 

blocked_found = 0; 

while ( z && (Z“>mark == D_NODE_BLOCKED |1 z~>type == D_N0DE_DIR) 
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{ if ( z->type != D_NODE_DIR) 

{ blocked_found = 1; 
z->parent = y; 
z->mark = D_NODE_UNBLOCKED; 
y->pert_child_count++; 
blocked_nodes_count— ; 

} 

go_to_sibling(z, 1) ; 

} 

if (blocked_found) blocked_chain_count--; 

} //to unblock a blocked chain in X“>link_one_side 

direction 

z = x->link_other_side; 

if ( z && (z->mark == D_NODE_BLOCKED | | Z’->type == D_NODE_DIR) ) 

{ 1 = x; 

blocked_found = 0; 

while ( z && (z->mark == D_N0DE_BL0CKED || z->type == D_N0DE_DIR) ) 
{ if ( z->type != D_NODE_DIR) 

{ blocked_found = 1; 

z->mark = D_N0DE_UNBLOCKED; 
z->parent = y; 
y->pert_child_count++; 
blocked_nodes_count--; 

} 

go_to_sibling(z, 1) ; 

} //to unblock a blocked chain in x->link_other_side 

direction 

if (blocked_found) blocked_chain_count--; 

} 


if ( iy ) 

root_reached = 1; 
else 

{ y->pert_child_count++; 

if (y“>mark == D_NODE_UNMARKED) 
{ Q.append(y); 

y->mark = D_NODE_QUEUED; 

} 

} 


} 

else 

{ // x*s parent is not valid 
k = x->link_one_side; 

1 = x; 

skip_dir(k,1); 

if (k->mark == D_NODE__BLOCKED) blocked_chain_count--; 
k = x->link_other_side; 

1 = x; 

skip_dir(k,1); 

if (k“>mark == D_NODE_BLOCKED) blocked_chain_count--“/ 

blocked_chain_count++/ 

blocked_nodes_count++; 

} 

} //end of "while (Q->size() + blocked_chain_count + root_reached > 1)" 
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if (blocked_chain_count) 

{ 

pseudo_root->pert_child_count = blocked_nodes_count; 
pseudo_root->type = D_NODE_QNODE; 

#ifdef _DEBUG_PQ_TREE 

// show("blocked_chain_count am ende von bubble",root); 
#endif 


} 

#ifdef _DEBUG_PQ_TREE 

//if (pseudo_root->leaf_index > 40) show("Ende bubble", root); 
#endif 

return true; 

} 


void pq_tree::bubble_reset(pq_node x) 

{ 

// in case bubble affects the nodes over the pertinent subtree root 
// we have to reset their pert_child_counts 

pq_node k; 

// while (x && x->pert_child_count) // alt 

while (x && x->pert_child_count && x != pseudo_root) 

{ 

if (x““>mark == D_NODE_UNBLOCKED) k= x->parent; 

else k= NULL; 
x->pert__child_count = 0; 
x->mark = D_NODE_UNMARKED; 
x->status= D_NODE_EMPTY; 

X = k; 

} 


bool pq_treereduce(list<int>& S) 

{ 

queue<pq_node> Q; 
pq_node x, y, k, 1; 

int S_size = S.sizeO; 

int x_type; 

//show_pq__tree_test (root) ; 

while (!S.empty()) 

{ X = leaves[S.pop ()]; 
x->pert_leaf_count = 1; 

Q.append(x); 
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} 


while (!Q.empty()) 

{ X = Q.pop(); 

#ifdef _DEBUG_PQ_TREE 
cons_pq_tree(root,"b","reduce") ; 
int aaa=0; 

#endif 

if (x->mark === D_NODE_BLOCKED) 

{ // a blocked chain exists and its members 
// get the "auxiliary" parent pseudo_root 

x->parent = pseudo_root; 
x->mark = D_NODE_UNBLOCKED; 

1 = x; 

k = x->link_one_side; 
skip_dir(k, 1); 

if (!k M k->mark == D_NODE_UNMARKED) 

{ if (pseudo_root->left_most) 

pseudo_root->right_most = 1; 
else 

pseudo_root->left_most = 1; 

} 


1 = x; 

k = x->link_other_side; 
skip_dir(k,1); 

if (!k II k->mark == D_NODE_UNMARKED) 

{ if (pseudo_root->left_most) 

pseudo_root->right_most = 1; 
else 

pseudo_root->left_most = 1; 

} 

if (Q.empty 0) pseudo_root->type = D_N0DE_QN0DE; 


} 

#ifdef _DEBUG_PQ_TREE 

//if ( ! aaa && x->type != D__NODE_LEAF) show("in reduce, der momentane 
Unterbaum",x); 

if (pseudo_root->leaf__index > 40) show ("in reduce, der gesamte Baum", 
root) ; 

cout << flush; 

#endif 

x_type= x->type; // Type may change in template 

application 

if (x->pert_leaf_count < S_size) 

{ 

if (x != pseudo_root) // ist immer true hier ? ! 

{ y = x->parent; 

y->pert_leaf_count += x->pert_leaf_count; 
if ( ! (—y->pert_child_count)) Q.append(y); 

} 
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if (x_type === D_IS10DE_LEAF) 

if (!template_PQLl(x, false)) return false; 

if (x_type == D_ISIODE_PNODE) 

if ( ! teinplate_PQLl (x, false)) 

if (!template_P3(x, false)) 

if (!template_P5(x, false)) return false; 

if (x_type == D_NODE_QNODE) 
if (!template_PQLl(x, false)) 
if (!template_Q2(x)) return false; 

} 

else 

{ // X is pruned pert subtree root 

(PRUNED(T, S)) 

pseudo_root->status = D_NODE_FULL; // ROOT(T,S) reached 

if (X“>parent) bubble_reset(x->parent); 

if (x_type == D_NODE_LEAF) 

if (!template_PQLl(x, true)) return false; 

if (x_type == D_NODE_PNODE) 
if (!template_PQLl(x, true)) 
if (!template_P3(x, true)) 
if (!template_P5(x, true)) 
if (!template_P6(x)) return false; 

if (x_type == D_NODE_QNODE) 
if (!template_PQLl(x, true)) 
if (!template_Q2(x) ) 
if ( !template_Q3(x) ) return false; 

} 

} // while Q not empty 

#ifdef _DEBUG_PQ_TREE 
cons_pq_tree(root,”b”,"reduce 2") ; 

//show("in reduce, der gesamte Baum nach Reduction", root); 

#endif 

return true; 

} 


void pq_tree::pert_sequence(list<int> &S) 

{ 

#ifdef _DEBUG_PQ_TREE 

if (!pseudo_root~>left_most){ 

cout << "pseudo_root->left_most gleich NULL in pert_sequence" << 
endl; 

exit(1); 

} 

#endif 

pq_node 1 = pseudo_root">left_most; 
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pq_node k; 

S.clear()/ 

S.push(O); // preparation 


if (1 == pseudo_root->right_most) 

//if pseudo_root has only one child life 

easy. 

{ sequence(S, 1); 
return; 


} 


is 


// Otherwise we have to find the direction to pseudo_root- 
>right_most. 

// Note that an endmost of pseudo_root not necessary has any 
NULL-link. 

if (l->link_one_side) 

{ k = l->link_one_side; 
skip_dir(k,1); 

1 = (k && k->status == D_NODE_FULL) 

? pseudo_root->left_inost->link_other_side 
: pseudo_root->left_most->link_one_side; 

} 

else 

1 = NULL; 

// The direction is detected, we can scan the sequence S. 
sequence(S, pseudo_root, 1); 

} 


void pq_tree::leaves_double() 


{ 

// 

array 

// 

of 

// 


// 

defining 


// 


if any leaf_index > leaves_size occurs we double the size of 

leaves, copy the content of the old array in the lower half 

the new and delete the old. Initially leaves_size is 16. 
Maybe the user had told the total number of leaves while 

his PQ tree. 


if (leaves_size) 

{ int i; 

pq_node* A = new pq_node[2 * leaves_size]; 

if (!A) error_handler(1,"pq_tree: out of memory”); 

for (i=0; i < leaves_size; i++) A[i] = leaves[i]; 
leaves_size *= 2; 

while (i < leaves_size) A[i++] = NULL; 
delete[] leaves; 
leaves = A; 

} 

else leaves init(32); 


return; 
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} 


void pq_tree::leaves_init{int Isize) 

{ 

leaves_size = Isize; 

leaves = new pq_node[leaves_size] ; 

if (Heaves) error_handler{1, "pq_tree: out of memory"); 
for (int i = 0; i < leaves_size; i+-f-) leaves [i] = NULL; 


void pq_tree::del_pert_subtree{) 

{ 

pq_node k = pseudo_root->left_most; 
pq_node 1 = NULL; 


while (k && k != pseudo_root->right_most) 

{ go_to_sibling(k, 1); 

del_subtree (1); // Note that also 1 is deleted in this function 
call. 

// But to progress correctly in the chain we need 

I’s 

// value (see previous command). 

} 


} 


// loescht alle Kinder von x und dann x selber: 

void pq_tree:;del_subtree{pq_node x) 

{ 

pq_node k == X“>left_most; 
pq_node 1 = NULL; 

while (k) 

{ //Note that the value of 1 is essentially for the loop 
//but the pointer 1 is not valid 
//{see also the comment in del_pert_subtree{) ). 

go_to_sibling{k, 1); 

del_subtree (1); //delete the subtree recursivly 

} 

delete x; 


} 


bool pq_treeupdate(list<int>& S) 

{ 

int i=0; 
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pq_node v, w, k= NULL, 1; 
if (S.empty()) 

{ // delete the pert, subtree contained in processed... 

while (!processed.empty()) 

{ w = processed.pop 0; 

if (w->type == D_NODE_LEAF) leaves[w->leaf_index] = NULL; 
delete w; 

} 


// if there’s anything else in the PQ_tree the reduction has 

failed. 


for(i=0; i < leaves_size && !leaves[i]; i++); 
return (i == leaves_size); 

} 


if (root && (pseudo_root->left_most == root)) 

{ // A special case requires special treatment. 

// Note that pseudo_root has only 1 child in this case. 

while (!processed.empty()) 

{ w = processed.pop 0 ; 

if (w->type == D_NODE_LEAF) leaves[w->leaf_index] = NULL; 
delete w; 

} 

// The pertinent subtree, here that means the whole pq_tree 
// is deleted. The application has failed, 
return false; 

} 

i = 0; 

V = new pq_node_struct; 

if (pseudo_root->left_most && 

pseudo_root“>left_most->parent_type == D_NODE_PNODE) 

{ V“>parent_type = D_NODE_PNODE; 

v->parent = pseudo_root“>left_most->parent; 

} 

else 

if (pseudo_root’->left_most ) V“>parent_type = D_NODE_QNODE; 
if (S.size () == 1) 

{ // then create v as a 

leaf 

v->leaf_index = S.popO; 

while (v->leaf_index >= leaves^size) leaves_double(); 
if (leaves[v“>leaf_index]) return false; 


leaves[V“>leaf_index] = v; 

V“>type = D_NODE_LEAF; 

} 

else 

{ //then create v as a P-node with leaves labelled with the elements 


// v->leaf_index = pseudo_root->leaf_index; 
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//only for testing 


v->type = D_NODE_PNODE; 

v->child_count = S.size(); 

v->left_most = 1 = new pq_node_struct; 

l->link_one_side = NULL; 

l->parent = v; 

l->leaf_index = S.pop(); 

while (l->leaf_index >= leaves_size) leaves_double(); 

if (leaves[l->leaf_index]) return false; 

leaves[l->leaf_index]=1; 
l->parent_type = D_NODE_PNODE; 
l->type = D_NODE___LEAF; 

while (!S.empty()) 

{ l->link_other_side = k = new pq_node_struct; 
k->leaf_index = S.popO; 

while (k->leaf_index >= leaves_size) leaves_double(); 

if (leaves[k->leaf_index]) return false; 

leaves[k->leaf_index] = k; 
k->parent_type = D_NODE_PNODE; 
k->type = D_NODE_LEAF; 
k->parent = v; 
k->link_one_side = 1; 

1 = k; 

} 

k->link_other_side = NULL; 
v->right_inost = k; 

} 

if (!root) 

{ //then the pq_tree is just constructed & still 

empty 

root = v; 

v->parent_type = D_NODE_QNODE; // *** warum ? 

return true; 

} 


//replace the full chain under pseudo_root by 

k = pseudo___root->lef t___most; 

1 = pseudo_root->right_most; 

if (k == 1) //then pseudo___root has only one child 

replace_in_siblings(k, v); 
else 
{ 

pq___node kl, 11; 

if (k->link_one_side && k->link___other_side) 

{ // Then k has a full and an empty sibling. 

// The full is contained in the pertinent subtree and to be 
// deleted. The empty becomes a sibling of v. 

if (k->link one side->status == D NODE EMPTY && 
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k->link:_one__side~>type != D_NODE_DIR) 
v->link_one_side = k->link_one_side; 

else 

if (k->link_other_side->status == D_NODE_EMPTY && 
k->link_other_side->type 1= D_NODE_DIR) 
v->link_one_side = k->link_other_side; 

else 

if ( k->link_one_side->status == D_NODE_FULL) 
V“>link_one_side = k->link_other_side; 
else 

if (k->link_other_side->status == D_NODE_FULL) 
v->link_one_side = k->link_one_side; 
else 

{ kl = k->link_one_side; 11 = k; 
skip_dir(kl,11); 
if (kl->status == D_NODE_EMPTY) 

v->link_one_side = k->link_one_side; 
else 

v->link_one_side = k->link_other_side/ 

} 

if (v->link_one_side->link_one_side == k) 
v->link_one_side->link_one_side = v; 

else 

v->link_one_side->link_other_side = v; 

} 

else 

{ // then k is an endmost of his real 

father 

v->link_one_side = NULL; 
if (k->parent->right_most == k) 
k->parent->right_most = v; 
else 

k->parent->left_most = v; 
v->parent = k->parent; 


// V is chained instead of k. 
if (l->link_one_side && l->link_other_side) 

{ // analogous to treatment of k, as above 

if (l->link_one_side->status == D_NODE_EMPTY && 
l->link_one_side->type ! = D_NODE__DIR) 
v->link_other_side = l->link_one_side; 

else 

if (l->link_other_side->status == D_NODE_EMPTY && 
l->link_other_side->type != D_NODE_DIR) 
v->link_other_side = l->link_other_side; 

else 

if (l->link_one_side->status == D_NODE_FULL) 
v->link_other_side = l->link_other_side; 
else 

if (l“>link_other_side->status == D_NODE_FULL) 
v->link_other_side = l->link_one_side; 
else 

{ kl = l->link_one_side; 11 = 1; 
skip_dir(kl,11); 
if (kl->status == D_NODE_EMPTY) 

v->link_other_side = l->link_one_side; 
else 
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v->link_other_side = l->link_other_side; 

} 

if (v->link_other_side->link_one_side == 1) 
v->link_other_side->link_one_side = v; 

else 

v->link_other_side->link_other_side = v; 

} 

else 

{ //then 1 is an endmost of his real father 

v->link__other_side = NULL; 
if (l->parent->right_most == 1) 
l->parent->right_inost = v; 
else 

l->parent->lef t_inost = v; 


v->parent = l->parent; 

} 

// V is chained instead of 1. 

} 

// Now V replaces the chain of full nodes beyond 
// the real father. 

// V is in the scanning direction of the pert subtree 
// and if necessary we can add a DIR-ptr 

if (pseudo_root->left_most->parent_type == D_NODE_QNODE) 

{ 

k = new pq_node_struct; 

k->leaf_index = pseudo_root->leaf_index/ 

k->type = D_NODE_DIR; // insert a DIR-ptr, but not as an 

end_inost 

if (v->link_one_side) 

{ k->link_one_side = v->link_one_side ; 
v->link_one_side = k; 
k->link_other_side = v; 
k->link_one_side->link_one_side == v ? 

(k->link_one_side->link__one_side = k) : 

(k->link_one_side->link_other_side = k); 

} 

else 

{ k->link_other_side = v->link_other_side ; 
v->link__other_side = k; 
k->link_one_side = v; 

k->link__other_side->link_one_side == v ? 

(k->link_other_side->link__one_side = k) : 

(k->link_other_side->link_other_side = k); 

} 

} 


// Prepare pseudo_root for the next possible DIR-ptr 

pseudo_root->leaf_index++; 
pseudo_root->mark = D_NODE_UNMARKED; 
pseudo_root->status = D_NODE_EMPTY; 

// Now we delete the pertinent subtree. 

while (!processed.empty 0) 
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{ w = processed.pop(); 

if (w->type == D_NODE_LEAF) leaves[w->leaf_index] = NULL; 
delete w; 

} 

#ifdef _DEBUG_PQ_TREE 

cons_pq_tree(root,"b","update"); 

#endif 

return true; 

} 


// h"aufig vorkommende Aktionen: 


inline void pq_tree;:skip_dir(pq_node& k, pq_node& last_k) 

// skip direction 

indicators 

{ 

while (k && k-'>type == D_NODE_DIR) 
go_to_sibling (k, last_k) ; 

} 


inline void pq_tree::go_to_sibling(pq_node& k, pq_node& last_k) 

{ 

if (k->link_one_side == last_k) 

{ 

last_k = k; 

k = k->link_other_side; 

} 

else 

{ last_k = k; 

k = k~>link_one_side; 

} 


165 


★★★★★★★★ 

+ 

+ LEDA 3.5.1 
+ 

+ _d2_spring.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

•k-k-k'kif-k'ki^'k'kif-k'k'k-k'k-k-k'k-k'k-k'k-k-k-k'k-k'k-k'k-k'k-k'k-k-kif'kif-k-kicifif-k-k-k-kic-k'kiri^'k'k'k'k'k'k'k'k'k-k'k-k-kif'k'k-k-k 
•k 'k -k -k -k -k -k y/ 

#include <LEDA/graph_alg.h> 

#include <math.h> 

#include <LEDA/array2.h> 


static float log_2(int x) 
{ float 1=0; 
while (x) 

{ 1 ++; 
x >>= 1; 

} 

return 1/2; 


void D2_SPRING_EMBEDDING(const graphs G, node_array<double>& xpos, 

node_array<double>& ypos, 
double xleft, double xright, 
double ybottom, double ytop, 
int iterations) 

{ double width = xright - xleft; 
double height = ytop - ybottom; 

for (int count = 1; count < iterations; count++) 

{ 

double k = sqrt(width*height / G.number_of_nodes()) / 2; 

//float 12 = 50*log_2(1+count); 

float 12 = 25*log_2(1+count) ; 

double tx = width / 12; 
double ty = height / 12; 

node_array<double> xdisp(G,0); 
node_array<double> ydisp(G,0); 

// repulsive forces 

node v; 
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forall_nodes(v,G) 

{ 

double XV = xpos[v]; 
double yv = ypos[v]; 

node u; 

forall_nodes(u,G) 

{ if(u — v) continue; 

double xdist = xv - xpos[u]; 
double ydist = yv - ypos[u]; 

double dist = xdist * xdist + ydist * ydist; 
if (dist < le-3) dist = le-3; 


double frepulse = k*k/dist; 


xdisp[v] 
ydisp[v] 

} 


+= frepulse xdist; 
+= frepulse * ydist; 


//xdisp[v] *= 

//ydisp[v] *= 


(double(rand_int(750,1250))/1000.0) 
(double(rand_int(750, 1250))/lOOO.O) 


// attractive forces 
edge e; 

forall_edges(e,G) 

{ node u = G.source(e); 
node V = G.target(e); • 

double xdist=xpos[v]“xpos[u]; 
double ydist=ypos[v]-ypos[u]; 

double dist=sqrt(xdist*xdist+ydist*ydist); 

float f = (G.degree(u)+G.degree(v))/16.0; 

dist /= f; 

xdisp[v]-=xdist*dist/k; 
ydisp[v]-=ydist*dist/k; 
xdisp[u]+=xdist*dist/k; 
ydisp[u]+=ydist*dist/k; 

} 


// preventions 

forall_nodes(v,G) 

{ double xd = xdisp[v]; 
double yd = ydisp[v]; 

double dist = sqrt(xd*xd+yd*yd) ; 

xd = tx*xd/dist; 
yd = ty*yd/dist; 

double xp = xpos[v] + xd; 
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double yp = ypos[v] + yd; 

//if (xp > xleft && xp < xright) 
xpos[v] = xp; 

//if (yp > ybottom && yp < ytop) 
ypos[v] = yp; 

} 

} 

} 


void D2_SPRING_EMBEDDING1(const graph& G, node_array<double>& xpos, 

node_array<double>& ypos, 
double xleft, double xright, 
double ybottom, double ytop, 
int iterations) 

{ double width = xright - xleft; 
double height = ytop - ybottom; 

for (int count = 1; count < iterations; count++) 

{ 

double k = sqrt (width^height / G. number__of_nodes () ) / 2; 

//float 12 = 50*log_2(1+count); 

float 12 = 25*log_2 (1+count); 

double tx = width / 12; 
double ty = height / 12; 

node_array<double> xdisp(G,0); 
node_array<double> ydisp(G,0); 

// repulsive forces 

node v; 

forall_nodes(V,G) 

{ double XV = xpos[v]; 
double yv = ypos[v]; 
node u; 

forall_nodes(u,G) 

{ if(u == v) continue; 

double xdist = xv ~ xpos[u]; 
double ydist = yv - ypos[u]; 

double dist = xdist * xdist + ydist * ydist; 

if (dist < le*“3) dist = le~3; 

double frepulse = k*k/dist; 

xdisp[v] += frepulse * xdist; 

ydisp[v] += frepulse * ydist; 

} 

edge e; 

forall_edges(e,G) 

{ node a = source(e); 
node b = target(e); 
if (a == V II b == v) continue; 
double xdist = xv - (xpos[a]+xpos[b])/2; 
double ydist = yv - (ypos[a]+ypos[b])/2; 
double dist = xdist * xdist + ydist * ydist; 
if (dist < le-3) dist = le-3; 
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double frepulse = k*k/dist; 
xdisp[v] += frepulse * xdist; 
ydisp[v] += frepulse * ydist; 

} 

} 


// attractive forces 
edge e; 

forall_edges(e,G) 

{ node u = G.source(e); 
node V = G.target(e); 

double xdist=xpos[v]-xpos[u]; 
double ydist=ypos[v]-ypos[u] ; 

double dist=sqrt(xdist*xdist+ydist*ydist) 

float f = (G.degree(u)+G.degree(v))/16.0; 

dist /= f; 

xdisp[v]-=xdist*dist/k; 
ydisp[v]-=ydist*dist/k; 
xdisp [u] -t-=xdist*dist/k; 
ydisp[u]+=ydist*dist/k; 

} 


// preventions 

forall_nodes(v,G) 

{ double xd = xdisp[v]/ 
double yd = ydisp[v]; 

double dist = sqrt(xd*xd+yd*yd)/ 

xd = tx*xd/dist; 
yd = ty*yd/dist; 

double xp = xpos[v] + xd; 
double yp = ypos[v] + yd; 

//if (xp > xleft && xp < xright) 
xpos[v] = xp; 

//if (yp > ybottom && yp < ytop) 
ypos[v] = yp; 

} 

} 

} 
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^ifif’kie'kirieififirieicie***'k**iciciri(i('k'kir'k'k-k'k-k'k'k*-k'k*'k'k'k'k'k'k-k'kie-k-k-k-k'k'k-k'kir'kir'kir'k'k'kir-k'k^ir^irir 

■k’k-k'k'k'k-k* 

+ 

+ LEDA 3.5.1 
+ 

+ _d3_spring.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

★★★★★★★/ 

#include <LEDA/graph_alg.h> 

#include <math.h> 

#include <LEDA/array2.h> 


static float log_2(int x) 
{ float 1 = 0; 
while (x) 

{ 1 ++; 

X >>= 1; 

} 

return 1/2; 


void D3_SPRING_EMBEDDING(const graphs G, node__array<double>& xpos, 

node_array<double>& ypos, 
node_array<double>& zpos, 
double xleft, double xright, 
double ybottom, double ytop, 
double zbottom, double ztop, 
int iterations) 

{ list<node> L; 

D3_SPRING_EMBEDDING(G,L,xpos,ypos,zpos, 
xleft,xright,ybottom,ytop,zbottom,ztop,iterations); } 


void D3_SPRING_EMBEDDING (const graphs G, const list<node>S fixed_nodes, 

node_array<double>S xpos, 
node_array<double>S ypos, 
node_array<double>S zpos, 
double xleft, double xright, 
double ybottom, double ytop, 
double zbottom, double ztop, 
int iterations) 

{ 

if (xleft >= xright I I ybottom >= ytop | | zbottom >= ztop) 
error_handler(1,"SPRING_EMBDDING: illegal bounds."); 
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double width = xright - xleft; 
double height = ytop - ybottom; 
double depth = ztop - zbottom; 

for (int count = 1; count < iterations; count++) 

{ 

double k = sqrt (width*height / G. nuiTiber_of_nodes () ) / 2; 

float 12 = 50*log_2(1+count); 

double tx = width / 12; 
double ty = height / 12; 
double tz = depth / 12; 

node__array<double> xdisp (G, 0) ; 
node_array<double> ydisp(G,0); 
node_array<double> zdisp(G,0); 

// repulsive forces 

node v; 

forall_nodes(v,G) 

{ int i = int((xpos[v] - xleft) / k) ; 
int j = int((ypos[v] - ybottom) / k) ; 

double XV = xpos[v]; 
double yv = ypos[v]; 
double zv = zpos[v] ; 

node u; 

forall_nodes(u, G) 

{ if(u == v) continue; 

double xdist = xv - xpos[u]; 
double ydist = yv - ypos[u]; 
double zdist = zv - zpos[u]; 

double dist = xdist * xdist + ydist * ydist + zdist * zdist; 
if (dist < le-3) dist = le-3; 


double frepulse = k*k/dist; 


xdisp[v] += 

ydisp[v] += 
zdisp[v] += 

} 

//xdisp[v] *= 

//ydisp[v] *= 
//zdisp[v] *= 

} 


frepulse * xdist; 
frepulse * ydist; 
frepulse * zdist; 

(double(rand_int(750,1250))/lOOO.0); 
(double(rand_int(750,1250))/lOOO.0); 
(double(rand int(750,1250))/1000.0); 


// attractive forces 
edge e; 

forall_edges(e,G) 

{ node u = G.source(e); 
node V = G.target(e); 

double xdist=xpos[v]-xpos[u] ; 
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double ydist=ypos[v]-ypos[u]; 
double zdist=zpos[v]-zpos[u]; 

double dist=sqrt(xdist*xdist+ydist*ydist+zdist*zdist) 
float f = (G.degree(u)+G.degree(v))/16.0; 
dist /= f; 

xdisp[v]-=xdist*dist/k; 
ydisp[v]-=ydist*dist/k/ 
zdisp[v]-=zdist*dist/k; 
xdisp[u]+=xdist*dist/k; 
ydisp[u]+=ydist*dist/k; 
zdisp[u]+=zdist*dist/k; 

} 


// preventions 

forall_nodes(v, G) 

{ double xd = xdisp[v]; 
double yd = ydisp[v]; 
double zd = zdisp[v]; 

double dist = sqrt(xd*xd+yd*yd+zd*zd); 

xd = tx*xd/dist; 
yd = ty*yd/dist; 
zd = tz*zd/dist; 

double xp = xpos[v] + xd; 
double yp = ypos[v] + yd; 
double zp = zpos[v] + zd; 

if (xp > xleft && xp < xright) xpos[v] = xp; 

if (yp > ybottom && yp < ytop) ypos[v] = yp; 

if (zp > zbottom && zp < ztop) zpos[v] = zp; 

} 

} 


} 
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★★★**★*★ 

+ 

+ LEDA 3.5.1 
+ 

+ _embedl.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 


// - 

// 

// straight line embedding 

// 

// K. Mehlhorn (1989) 

// - 


#include <LEDA/graph_alg.h> 


const int A = -2; 
const int B = -1; 

static node_array<list_item> Classloc; 
static node_array<int> ord, labelled. Class; 
static node_array<node> first, second, last; 

void label_node(graphs G, list<node>& L, int& count, 

list<node>& Al,list<node>& Bl, list<node>*& II, 
node V, node c) 

{ // labels the node v; c is the special node which is to be labelled 
// last; the details are described in lemma 10 
edge e; 

L.append(v); 
ord[v]=count++; 
labelled[v]=1; 

forall_adj_edges(e, v) 

{ edge el = G.reversal(e) ; 
node tt = target(e); 
int i; 

if (labelled[tt] && !labelled[target(G.cyclic_adj_succ(e))]) 

{ first[v]=tt; 

second[v]=target(G.cyclic_adj_pred(e)); 

} 
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if (labelled[tt] && !labelled[target(G.cyclic_adj_pred(e))]) 
last[v]=tt; 

if (!labelled[tt] && (tt != c)) 

{ if (Class[tt] == A) 

{ A1.del(Classloc[tt]) ; 

Classloc[tt] = Bl,push(tt); 

Class[tt]=B; 

} 

else 

{ if (Class[tt] == B) 

{ B1.del(Classloc[tt]); 

i = 2-labelled[target(G.cyclic_adj_succ(el))] 
-labelled[target(G.cyclic_adj_pred(el))]; 

} 

else 

{ i=Class[tt]; 

II[i].del(Classloc[tt]); 

i = i+l-labelled[target(G.cyclic_adj_succ(el))] 
-labelled[target (G. cyclic_adj__pred(el) ) ] ; 

} 

Class[tt]-i; 

Classloc[tt]=11[i]-push(tt); 

}//end else case 
}//end if 
}//end 

}//end of label node 


void compute_labelling(graphs G,list<node>& L, list<node>& Pi) 

{ /* computes the ordering of lemma 10 in List L ,the sequence pi 

in List Pi, the function L'^-l (v) in Array ord, and the functions 
first,second,last of lemma 11 in the corresponding Arrays 

*/ 

node v,a,b,c; 

/* zuerst berechne ich die drei Knoten,die am Rand des aeusseren 
Gebiets liegen sollen 

*/ 

a=G. first__node () ; 

list<edge> temp = G.adj_edges(a); 
b = target(temp.pop()); 
c = target(temp.pop 0); 


/* 

node_array<int> labelled(G,0); 

*/ 

labelled.init(G,0); 
int count =0; 
list<node> A1 ; 
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node_array<int> Class(G); 
node_array<list_item> Classloc(G); 

*/ 


Class.init(G); 

Classloc.init(G); 

forall_nodes(V,G) { Classloc[v] = A1.push(v);Class[v]=A;} 
list<node> Bl; 

list<node>* II = new list<node>[G.number_of_nodes()]; 

label_node(G,L,count,Al,Bl,II,a, c) ; 
label_node(G,L,count,Al,Bl,II,b,c); 

while ( !II[1].empty() ) 

{ node V = Il[l].pop(); 

label_node (G, L, count, Al, Bl, II, v, c) ; 

} 

label_node(G,L,count,Al,Bl,II,c,c); 

//nun berechne ich noch first second und last des Knoten c 
first[c]=a; 
last[c]=b; 

edge e; 

forall__adj_edges (e, c) if (target (e) ==a) 
second [c] =target (G . cyclic_adj_pred (e) ) ; 

//nun die Folge Pi 
node_array<list_item> Piloc(G); 

Piloc[a] = Pi.push(a)/ 

Piloc[b] = Pi.append(b); 

forall(v,L) if (v != a && v != b) Piloc[v] = 

Pi.insert(v,Piloc[second[v]],-1); 

}//end of compute_labelling 

void move_to_the_right(list<node>& Pi, node v, node w, 

node_array<int>& ord, node_array<int>& x) 

{ //increases the x-coordinate of all nodes which follow w in List Pi 
//and precede v in List L,i.e., have a smaller ord value than v 
int seen_w = 0; 
node z; 
forall(z,Pi) 

{ if (z==w) seen_w=l; 

if (seen_w && (ord[z]<ord[v])) x[z]=x[z]+l; 

} 

} 


int STRAIGHT_LINE_EMBEDDING(graphs G,node_array<int>S x, 
node_array<int>S y) 

{ 

// computes a straight-line embedding of the planar map G into 
// the 2n by n grid. The coordinates of the nodes are returned 
// in the Arrays x and y. Returns the maximal coordinate. 
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if (G.emptyC)) return 0; 

if (G. nuinber_of_nodes () == 1) 

{ node V = G.first_node(); 
x[v] = y[v] = 1; 
return 1; 

} 

list<node> L; 
list<node> Pi; 
list<edge> TL; 

node v; 
edge e; 

int maxcoord = 1; 

/* 

node_array<int> ord(G); 

node_array<node> first(G), second(G), last(G); 
*/ 

ord.init(G); 
first.init(G); 
second.init(G); 
last.init(G); 


TL = G.triangulate^map(); 
if (!G.make_map()) 

error_handler (1, "STRA.IGHT LINE EMBEDDING: graph must be a planar 
map”); 


compute_labelling(G,L,Pi); 


//I now embed the first three nodes 

V = L.pop(); 
x[v] = 0; 
y[v] = 0; 

if (!L.empty()) 

{ V = L.pop(); 
x[v] = 2; 
y[v] = 0; 

} 

if (!L.empty()) 

{ V = L.pop(); 
x[v] = 1; 
y[v] = 1; 

} 


//I now embed the remaining nodes 

while ( !L.empty() ) 

{ 

V = L.pop(); 
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// I first move the nodes depending on second[v] by one unit 
// and the the nodes depending on last[v] by another unit to the 
// right 


move_to_the_right(Pi,v,second[v],ord,x); 
move_to_the_right(Pi,v,last[v],ord,x) ; 

// I now embed v at the intersection of the line with slope +1 
// through first[v] and the line with slope -1 through last[v] 

int x_first_v = x[first[v]]; 
int x_last_v = x[last[v]]; 
int y_first_v = y[first[v]]; 
int y_last_v = y[last[v]]/ 

X [v] = (y_last_v - y_first_v + x_first_v + x_last_v)/2; 
y [v] = (x_last_v - x_first_v + y_first_v + y_last_v)/2; 


// delete triangulation edges 
forall(e,TL) G.del_edge(e) ; 

forall_nodes(V,G) maxcoord = Max(maxcoord,Max(x[v],y[v])); 
return maxcoord; 


void STRAIGHT_LINE_EMBEDDING(graphs G,node_array<double>& x, 
node_array<double>& y) 

{ 

node_array<int> xO(G); 
node_array<int> yO(G); 

int maxc = STRAIGHT_LINE_EMBEDDING(G, xO, yO) ; 
node v; 

forall_nodes(v,G) 

{ x[v] = double(xO[v])/maxc; 
y[v] = double(yO[v])/maxc; 

} 
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+ 

+ LEDA 3.5.1 
+ 

+ _einbed2 . c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104) . 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

★*★★★★★j 


// - 

// Dirk Ambras 1995 
//- 


#include <LEDA/graph_alg.h> 

static void Contract(graphs G, node a, node b , node c, list<node>& L) 

{ 

node v,w; 

list<node> cand; 

node_array<bool> marked(G, false); // betrachtete Knoten 

node_array<int> deg(G,0); // # virtuelle Nachbarn 

int N = G.number_of_edges() ; 

marked[a] = marked[b] = marked[c] = true; // Init 

deg[a] = deg[b] = deg[c] = N; 

forall_adj_nodes(v,a) 

{ marked[v]=true; 

forall_adj_nodes(w,v) deg[w]++; // mache v bekannt bei den w*s 

} 


{ forall_adj_nodes(V,a) // lade Kandidaten 

if (deg[v]<=2) cand.append(v); 

} 


while (!cand.empty()) 

{ node u=cand.pop(); 
if (deg[u] == 2) 

{ L.push(u); 
deg[u]=N; 

forall_adj_nodes(v, u) 

{ deg[v]--; // u ist virtuell geloescht 

if (!marked[v]) // v ist neuer Nachbar von a 

{ marked[v]=true; 
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forall_adj_nodes(w,v) deg[w]++; 

den w*s 

if (deg[v] <= 2) cand.append(v) ; 
} else 

if (deg[v] == 2) cand.append(v); 

} 

} 

} 

} 


// mache v bekannt bei 
// lade Kandidaten 


static void Realizer(graphs G, const list<node>& L, 

node a, node b, node c, 

GRAPH<node, int>& T, node_array<node>& v_in_T) 

{ 

int i=0; 
node v; 
edge e; 

node_array<int> ord(G,0); 

ord[b] = i++; 
ord[c] = i++; 
node u; 

forall(u,L) ord[u]=i++; // V(G) numerieren 

ord[a] = i-i-+; 

forall_nodes(v, G) v_in_T[v] = T.new_node() ; // T = copy of G 

forall(v, L) 

{ node u = v_in_T[v]; // u is copy of v in T 

forall_adj_edges(e, v) 

if (ord[G.target (e)] > ord[v]) break; 

edge el = e; 

while(ord[G.target (el)] > ord[v]) el = G.cyclic_adj_succ(el)/ 
T.new_edge(v_in_T[G.target(el)], u, 2) ; 

edge e2 = e; 

while(ord[G.target (e2)] > ord[v]) e2 = G.cyclic_adj_pred(e2); 

T.new_edge(v_in_T[G.target(e2)], u, 3); 

for(e=G.cyclic_adj_succ(el); e != e2; e=G.cyclic_adj_succ(e)) 

T.new_edge(u, v_in_T[G.target(e) ] , 1) ; 

} 

// special treatement of a,b,c 

node a_in_T = v_in_T[a]; 
node b_in_T = v_in_T [b] ; 
node c_in_T = v_in_T[c]/ 

forall_adj_edges(e,a) 


T.new edge(a in T, 

V in_T[G 

.target(e)], 1); 

T.new_edge(b_in_T, 

a_in_T, 

2) ; 

T.new edge(b in T, 

c__in_T, 

2) ; 

T.new edge(c in T, 

a in T, 

3) ; 

T.new edge(c in T, 

b_in_T, 

3) ; 
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} 


static void Subtree_Sizes(GRAPH<node, int>& T, int i, node r, 

node_array<int>& size) 


// computes sizes of all subtrees of tree with root r in T(i) 


int sum=0/ 
edge e; 

forall_adj_edges(e, r) 
if (T[e]==i) 

{ node w=T.target(e) ; 

Subtree^Sizes(T, i, w, size); 
sum+=size[w]; 

} 

size[r]=sum+l; 


static void 

{ 


Prefix_Sum(GRAPH<node, int>& T, int i, 
const node_array<int>& val. 


node r, 

node_array<int>& sum) 


// computes for every node u in the subtree of T(i) with root r 
// the sum of all val[v] where v is a node on the path from r to u 

list<node> Q; 

Q,append(r)/ 
sum[r] = val[r]; 

while (!Q.empty()) 

{ node v=Q.pop(); 
edge e; 

forall__adj_edges (e, v) 
if (T[e] == i) 

{ node w=T.target(e); 

Q.append(w)/ 

sum[w] = val[w] + sum[v]; 

} 

} 

} 


int STRAIGHT_LINE_EMBEDDING2(graphs G,node_array<int>& xcoord, 

node_array<int>& ycoord) 


int n = G.number of nodes(); 


if (n < 3) 


{ int max_c =1; 
if (n > 0) 

{ node a = G.first_node(); 
xcoord[a] = 1; 
ycoord[a] = 1; 


if (n > 1) 

{ node b = G.last_node() ; 
xcoord[b] = 2; 
ycoord[b] =2; 
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max_c = 2; 

} 

return max c; 


node v; 

list<node> L; 

GRAPH<node, int> T; 
node_array<node> v_in_T(G); 


list<edge> el = G.triangulate_map(); 
// choose outer face a,b,c 


node a=G.first_node(); 

edge e=G.first_adj_edge(a); 

node c=G.target(e)/ 

node b = G.target(G.adj_succ(e) ) ; 


Contract(G,a,b,c,L); 


Realizer(G,L,a,b,c,T,V in T); 


// T aufbauen 


node_array<int> 

node_array<int> 

node_array<int> 


tl(T); 

t2(T); 

val(T,l); 


node_array<int> PI(T); 
node_array<int> P3(T); 
node_array<int> vl(T); 
node_array<int> v2(T); 


Subtree_Sizes(T, 
Subtree Sizes(T, 


1, v_in_T[a], tl); 

2, V in T[b], t2); 


Prefix_Sum (T, 
Prefix_Sum(T, 
// now Pi = 


1, v_in_T[a], val, PI); 

3, v_in_T[c], val, P3)/ 

depth of all nodes in Tree T(i) (depth[root] 


1 ) 


Prefix_Sum(T, 2, v_in_T[b], tl, vl); 

vl[v_in_T[a]] = 11[v_in_T[a]] ; // Sonderrolle von a 

// in vl[v] steht jetzt die Summe (Anzahl der Knoten im Tl-UBaum[x]) 

// fuer jeden Knoten x im Pfad in T2 von b nach v 

Prefix_Sum(T, 3, v_in_T[c], tl, val) ; 

val[v in T[a]]=tl[v in T[a]]; // Sonderrolle von 


// in val[v] steht jetzt die Summe (Anzahl der Knoten im Tl-UBaum[x]) 
// fuer jeden Knoten x im Pfad in T3 von c nach v 
// es ist rl [v] =vl [v]+val [v]--tl [v] die Anzahl der Knoten in der 
// Region 1 von v 

forall^nodes(v, T) vl[v] += val[v]-t1[v]-P3[v]; // vl* errechnen 

Prefix_Sum (T, 3, v_in_T[c], t2, v2) ; 
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v2 [ v_in_T [b] ] =t2 [ v_in_T [b] ] / 
b 


// Sonderrolle von 


Prefix_Sum(T, 1, v_in_T[a], t2, val)/ 

val[v_in_T[b]]=t2[v_in_T[b]]; // Sonderrolle von 

b 

forall_nodes(v, T) v2[v] += val[v]-12[v]-PI[v]; // v2’ errechnen 

int maxcoord = 0; 

forall_nodes(v, G) // x- & y-Feld kopieren 

{ xcoord[v] = vl[v_in_T[v] ] ; 
ycoord[v] = v2[v_in_T[v]]; 

maxcoord = Max(maxcoord, Max(xcoord[v],ycoord[v])); 

} 

forall(e, el) G.del_edge(e); // 

eingefuegte Kanten 

// loeschen 

return maxcoord; 

} 


void STRAIGHT_LINE_EMBEDDING2(graphs G,node_array<double>& x, 
node_array<double>& y) 

{ 

node_array<int> xO(G); 
node_array<int> yO(G); 

int maxc = STRAIGHT_LINE_EMBEDDING2(G,xO,yO); 
node v; 

forall_nodes(v,G) 

{ x[v] = double(xO[v])/maxc; 
y[v] = double(yO[v])/maxc; 

} 
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+ 

+ LEDA 3.5.1 
+ 

+ _ortho.c 
+ 

-I- This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 


#include <LEDA/planar_map.h> 
ttinclude <LEDA/node_map.h> 
tinclude <LEDA/edge_map.h> 
#include <LEDA/face_map.h> 
#include <LEDA/stack.h> 
#include <LEDA/graph_alg.h> 


#define EPS 

tdefine INFINITY MAXINT 
ttdefine REV{e) P.reversal(e) 
tdefine SUCC(e) P.face_cycle_succ(e) 
tdefine PRED(e) P.face_cycle_pred(e) 

tdefine IS_CAGE(f) (P.get_cage(P.first_face_edge(f))) 
tdefine IS_OUTER(f) {!P.get_inner(P.first_face_edge{f))) 

tdefine NEXT(d) ((direction)((d+5)%4)) 

tdefine PREV(d) ((direction)((d+3)%4)) 
tdefine OPP(d) ((direction)((d+6)%4)) 


tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 

tdefine 


ERR_EMPTY_GRAPH 

ERR_NOT_CONNECTED 

ERR_NO_PLANAR_MAP 

E RR_INVALID_NETWORK 

ERR_NO_FEASIBLE_FLOW 

ERR_NO_ORTHO_REP 

ERR_BAD_STRING 

ERR_BAD_ANGLE 

ERR_BAD_DIRECTION 

ERR_OPEN_CPLEX_ENV 

ERR_LOAD_LP 

ERR_CPLEX_ADD_ROWS 

ERR_CPLEX_WRITE 

ERR_CPLEX_OPT 

ERR_CPLEX_GET_SOLUTION 

ERR_CPLEX_FREE 

ERR CPLEX CLOSE 


"ORTHO: 

"ORTHO: 

"ORTHO: 

"ORTHO: 

"ORTHO: 

"ORTHO: 

"ORTHO: 

"ORTHO: 

"ORTHO: 

"CPLEX: 

"CPLEX: 

"CPLEX: 

"CPLEX: 

"CPLEX: 

"CPLEX: 

"CPLEX: 

"CPLEX: 


input graph is empty" 

input graph is not connected" 

this is no planar map" 

invalid network" 

no feasible flow" 

orthogonal rep. not valid" 

trying to set nonempty string" 

bad angle" 

bad direction" 

failed to open environment" 

failed to load LP" 

CPXaddrows failed" 

CPXwriteLP failed" 
failed to optimize LP" 
failed to obtain solution" 
CPXfreeprob failed" 
CPXcloseCPLEX failed" 


enum direction{north,east,south,west,unexplored}; 
enum v^type{real,bend,dissection,big}; 


void longest_paths(const GRAPH<int,int>&G, node_array<int>&l){ 
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node_array<int>INDEG(G,0);node v;edge e; 
stack<node>S; 
forall_nodes(v,G){ 

INDEG[v]= indeg(v); 
if(INDEG[v]==0)S.push(v); 

} 

while(!S.empty()){ 
v= S.pop(); 

forall_out_edges(e,v){ 
node w= target (e); 
l[w]= Max(l[w],l[v]+G.inf(e)); 
if ( —INDEG [w] ==0) S .push (w) ; 

} 

} 

} 


int angle(const char c){ 
int result= 0; 
switch(c){ 

case’O’:result= 90;break; 
case’1 *:result= 27 0/break; 
default:error_handler(1,ERR_BAD_ANGLE); 
} 

return result; 

} 


bool Euler(const graph&G){ 

int n= 0;node v;forall_nodes(v,G)if(outdeg(v)!=0)++n; 
int m= G.number_of_edges()/2; 
int f= G.number_of_faces0; 
return(m==n+f~2); 

} 


class ortho_map:public planar_map{ 
node_map<node>v_in_G; 
node_map<v_type>v_T; 
edge_map<edge>e_in_G; 
edge_map<edge>e_in_P; 
edge_map<int>a; 
edge_map<string>s; 
edge_map<bool>inner; 
edge_map<bool>cage; 
edge_array<direction>dir; 
edge_array<int>length; 
node_array<int>x, y; 

public; 

edge next_level_edge(const node,const direction); 
ortho_map(const graphs); 
void print(); 

//bool check 0; 

void init_maps(const edge,const int,const string,const bool,const bool); 
node get_orig(const node v)const{return v_in_G[v];} 
edge get_orig(const edge e)const{return e_in_G[e];} 
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direction get_dir(const edge e)const{return dir[e]/} 

edge get_copy(const edge e)const{return e_in_P[e];} 

v_type get_type(const node v)const{return v_T[v];} 

void set_inner(const edge e,const bool t){inner[e]= t;} 

void set_cage(const edge e,const bool t){cage[e]= t;} 

bool get_inner(const edge e)const{return inner[e]/} 

bool get_cage(const edge e)const{return cage[e];} 

void set_type(const node v,const v_type t){v_T[v]= t;} 

int get_a(const edge e)const{return a[e];} 

void set_a(const edge e,const int angle){a[e]^ angle;} 

string get_s(const edge e)const{return s[e];} 

void set_s(const edge e,const string s_new){s[e]= s_new;} 

void set_s(const edge e,const int,const int,const bool); 

int get_x(const node v)const{return x[v];} 

int get_y(const node v)const{return y[v];} 

void set_rev_s(const edge e,const int,const int,const bool); 

void set_length(const edge e,const int l_new){length[e]= l_new;} 

int get_length(const edge e)const{return length[e];} 

edge split_map_edge(edge) ; 

edge new_edge(edge,edge) ; 

edge split_bend_edge(edge); 

void PrintEdge(const edge); 

void init_rest(){dir.init(*this,unexplored); 

length.init(*this,0); 

x.init(*this,-1);y.init(*this,-1);} 


void ortho_map::assign_directions(edge e,direction d){ 
while(dir[e]==unexplored){ 
dir[e]= d;edge r= reversal(e); 
if(dir[r]==unexplored){ 

if(inner[r])assign_directions(r,OPP(d)); 
else dir[r]= OPP(d); 

} 

switch(a[e]){ 
case 90:d= NEXT(d);break; 
case 270:d= PREV(d);break; 
case 360:d= OPP(d);break; 

} 

e= face_cycle_succ(e); 

} 

} 

void determine_position(const node,const int x,const int 

y,node_array<bool>&); 

void norm_positions(); 

edge succ_corner_edge(const edge); 

'-ortho_map () {clear () ; } 

LEDA_MEMORY(ortho_map) 

}; 


edge ortho_map::next_level_edge(const node v,const direction d){ 
edge e; 
switch(d){ 

case east: { forall_out_edges(e,v) if(dir[e]==north) return e; break; 

case west: { forall_in_edges(e,v) if(dir[e]==north) return e; break; 

case north:{ forall in edges(e,v) if(dir[e]==east) return e; break; 


} 

} 

} 
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case south:{ forall_out_edges(e,v) if(dir[e]==east) return e; break 
default:break; 

} 

return NULL; 

} 


ortho_inap: : ortho_map (const graph&G) ;planar_inap (G) { 

if (G. number_of_nodes () ==0) error_handler (1, ERR_EMPTY_GRAPH) ; 

if ( ! Is_Connected (G) ) error_handler (1, ERR_NOT_CONNECTED) ; 

if ( ! Euler (G) ) error_handler (1, ERR_NO_PLANAR_MAP) ; 

v_in_G. init (*this ) ; e_in_G. init (*this) ; 

e in P.init(G); 


node v_P= first_node(),v_G= G.first_node() ; 

while (v_P) { 

v_in_G[v_P]= v_G; 

v_P= succ_node(v_P); 

v_G= G.succ_node(v_G); 

} 

forall_nodes(v_P, *this) { 
v_G= v_i n_G [ v_P ] ; 

edge e_P= f irst_adj_edge (v_P) , e_G= G. f irst_adj_edge (v_G) ; 
while (e_P) { 

e_in_G [ e_P ] = e_G; e_in_P [ e_G] = e_P ; 
e_G= adj_succ (e_G) ; e_P= adj_succ (e_P) ; 

} 

} 


t 

a . init (*this, 90) ; s . init (*this, EPS) ; v_T . init (*this, real) ; 
inner. init (^^^this , true) ; cage. init (*this, false) ; . 

} 


edge ortho_inap: : split_inap_edge (edge e) { 
edge n= planar_map::split_edge(e) ; 
edge er= reversal(e),nr= reversal(n); 
a[er]= a[nr];a[n]= a[e]; 
s[er]= EPS;s[n]= EPS; 

inner[er]= inner[nr];cage[er]= cage[nr]; 
inner[n]= inner[e];cage[n]= cage[e]; 
e_in_G[n]= e_in_G[e]; 
e_in_G[er]= e_in_G[nr] ; 
v_in_G[source(n)]= NULL; 

return n; 

} 

edge ortho_map: : split_bend__edge (edge e) { 

string s_e= s[e];int a_e= a[e]; 

bool inner_e= inner[e];bool cage_e= cage[e]; 

edge er= reversal(e); 

string s_er= s[er];int a_er= a[er]; 
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bool inner_er= inner[er];bool cage_er= cage[er]; 

edge n= split_map_edge(e); 

er= reversal(e);edge nr= reversal(n); 

a[e]= angle(s_e[s_e.length 0-1]);a[er]= a_er; 
a[n]= a__e;a[nr]= angle (s_er [0] ) ; 

s[e]= s_e.head(s_e.length()-1);s[er]= s_er.tail(s_er.length()-1); 

s[n]= EPS;s[nr]= EPS; 

inner[e]= inner[n]= inner_e; 

cage[e]= cage[n]= cage_e; 

inner[er]= inner[nr]= inner_er; 

cage[er]= cage[nr]= cage_er; 

v_T[source(n)]= bend; 

return n; 

} 

edge ortho_map::new_edge(edge el,edge e2){ 
edge n= planar_map::new_edge(el, e2) ; 
e_in_G[n]= NULL; e_in_G [reversal (n) ] = NULL; 
return n; 

} 


void ortho__map: : print 0 { 
face f;edge e; 
forall_faces(f,(*this)){ 

if(!inner[first_face_edge(f)])cout<<”outer ”; 
if(cage[first_face_edge(f)])cout<<"cage "; 
cout<<”face: "<<endl; 

forall_face_edges(e,f){PrintEdge(e);newline; } 

} 

} 

/* 

bool ortho_map::check() { 
bool result= true;return result; 
face f;edge e;node v; 
forall_faces(f,(*this)){ 
int rotation= 0; 

forall_face_edges(e,f)rotation+= zeroes(s[e])-ones(s[e])+2-a[e]/90; 
if(!inner[first_face_edge(f)])result= result&&(rotation==-4); 
else result= result&&(rotation==4); 

} 

forall_nodes(v,(*this))if(outdeg(v)!=0){ 
int angle_sum= 0; 

forall_in_edges (e, v) angle_sum+= a [e] ; 
result= result&& (angle_suin==360) ; 

} 

return result; 

} 

*/ 

void ortho_map::set_s(const edge e,const int flow,const int flow_rev, 
const bool bridge){ 

if(s[e] !=EPS)error_handler(1,ERR_BAD_STRING) ; 
for(int i= 0;i<flow;i++)s[e]+= ”0”; 
if(Ibridge){ 

for(int i= 0;i<flow_rev;i++)s[e]+= ”1”; 

} 


187 


} 


void ortho_map::set_rev_s(const edge e,const int flow,const int 
flow_rev, 

const bool bridge){ 

if(s [e] !=EPS)error_handler(1,ERR_BAD_STHING); 
for(int i= 0/i<flow;i++)s[e]+= "1”; 
if(Ibridge){ 

for(int i= 0/i<flow_rev;i++)s[e]= string(*0*)+s[e]; 

} 

} 

void ortho_map::PrintEdge(const edge e){ 
planar_map::print_edge(e); 

cout<<"\t (**<<s [e] «” ”«a [e] <<" [ "<<inner [e] «cage [e] <<” ] ) "; 

} 

void ortho_inap :: init_inaps (const edge e, const int a_new, const string 
s_new, 

const bool i_new,const bool c_new){ 

a[e]= a_new;s[e]= s_new; 

inner[e]= i_new;cage[e]= c_new; 

} 


void ortho^map:: deterniine_position (const node v, const int x_new, 

const int y_new,node_array<bool>&seen){ 

if(seen[v])return; 

edge e;x[v]= x_new/y[v]= y_new; 

seen[v]= true; 

forall_out_edges(e,v){ 

if(dir[e]==north) 

determine_position(target(e),x_new+length[e] , y_new, seen) ; 
if(dir[e]==west) 

deteririine_position (target (e) , x_new, y_new+length [e] , seen) ; 

} 

forall_in_edges(e, v) { 
if(dir[e]==north) 

determine^position(source(e),x_new-length[e],y_new, seen); 
if(dir[e]==west) 

determine_position(source(e),x_new,y_new“length[e] , seen); 

} 

} 

void ortho^map::norm^positions(){ 
int xniin= 0,yinin= 0;node v; 
forall_nodes(v,(*this))if(v_T[v]!=big){ 
xniin= Min (xmin, X [ v] ) ; 
yinin= Min (ymin, y [ v] ) ; 

} 

forall_nodes(v,(*this)){ 
x[v]-= xmin; 
y[v]-= ymin; 

} 

} 

edge ortho_map::succ_corner_edge(const edge e){ 
edge e_c; 

for(e_c= face_cycle_succ(e);a[e_c]==180;e_c= face_cycle_succ(e_c)); 
return e_c; 

} 
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/* 

bool Euler(const graphs); 
int angle(const char); 
int zeroes(const strings); 
int ones(const strings); 

void longest_paths(const GRAPH<int,int>S,node_array<int>S) 


#include ’’common.h” 
#include ”ortho_map.h" 


#include 

#include 

#include- 

#include 

#include 


<LEDA/stack.h> 
<LEDA/array.h> 

<LEDA/set.h> 
<LEDA/integer_matrix.h> 
<LEDA/graph_alg.h> 


#ifdef CPLEX 
extern"C” 
#include<cplex.h> 
#endif 


typedef list<int> intlist; 

typedef list<node> nodelist; 

int ORTHO_EMBEDDING(const graphSG, 

node_array<int>Sx_pos, 
node_array<int>Sy_pos, 
edge_array<intlist>Sx_bends, 
edge_array<intlist>Sy_bends, bool ) 

{ 

ortho_map P(G); 


node_map<edge>corr_cage_edge(P); 
edge_array<nodelist>b_nodes(G); 
edge_array<node>b_nodes_first(G,NULL); 
edge_array<node>b_nodes_last(G,NULL); 
list<node>all_nodes= P.all_nodes(); 
node v;edge e;face f; 
forall(V,all_nodes)if(outdeg(v)>4){ 

P.set_type(v,big); 
int d= outdeg(v),i= 0; 
array<edge>out(d);edge e_cage; 
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forall_out___edges (e, v) { 

out[i++]= e;edge e_orig= P.get_orig(e); 
edge e_split= P.split_map_edge(e); 
node c_i= source{e_split); 

P.set_type(c_i,bend); 
b_node s_fir s t[e_orig]= c_i ; 
b_nodes_last[G.reversal{e_orig)]= c_i; 

} 


for(i= 0;i<d;i++){ 

e_cage= P.new_edge(SUCC(out[i]),REV(out[(i + l)%d] ) ) ; 
P.init_maps(e_cage,90,EPS,true,true) ; 
edge r_cage= P.reversal(e_cage) ; 

P.init_maps(r_cage,90,EPS,true,false) ; 

} 


corr_cage_edge[v]= e_cage; 
forall_out_edges(e,v)P.join_faces(e) ; 


} 

//cout<<'*cages created"«endl; 


//if(check)if(!Euler(P))error_handler(l,ERR NO PLANAR MAP) 


P.compute_faces(); 
face f_0;int f_0_deg= 0; 
forall_faces(f,P) 

if(!IS_CAGE(f)&&P.size(f)>f_0_deg) { 
f_0= f;f_0_deg= P.size(f_0); 

} 

forall_face_edges (e, f_0) P. set_inner (e, false) ; 


graph N; 
node s, t; 

edge_array<int>cap,cost, 1; 
int z= 0; 


list<node>V_hat; 
list<node>F/ 
node_map<node>NtoV(N) ; 
node_map<face>NtoF(N) ; 
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face_map<node>FtoN(P); 


s= N.new_node{)/ 
t= N.new_node{); 

forall_nodes(v,P)if(outdeg(v)>0&&outdeg(v) <=3) { 
node n= N.new_node(); 

NtoV[n]= v; 

V_hat.append(n); 

} 

forall_faces(f,P){ 
node n= N.new_node(); 

NtoF[n]= f;FtoN[f]= n; 

F.append(n); 

} 


list<edge>A_s_v, A_s_f, A_v, A_v_cage, A_f, A_f_t ; 


forall_faces(f,P){ 

int size= P.size(f); 

if{!IS_OUTER(f)&&size<=3) 

A_s_f.append(N.new_edge(s,FtoN[f] ) ) ; 

if(IS_OUTER(f) I Isize>=5)A_f_t.append{N.new_edge{FtoN[f] , t) ) 

} 

forall{V,V_hat)A_s_v.append(N.new^edge(s, v) ) ; 


forall{V,V_hat){ 
set<face>F_adj; 

forall_adj_faces(f,NtoV[v])F_adj.insert(f) ; 
forall(f,F_adj) 

if (IS_CAGE (f) ) A_v_cage . append(N.new_edge (v, FtoN [f ] ) ) / 
else A_v.append(N.new_edge(v,FtoN[f])); 

} 


#define MAX_BENDS_PER_EDGE INFINITY 
edge_array<bool>marked(P, false) ; 
edge_map<edge>partner(N),a_in_P(N);edge al,a2; 
forall_faces(f,P) 

forall_face_edges(e,f)if{!marked[e]){ 
face g= P.face_of(REV(e)); 
if{f==g){ 

al= N.new_edge(FtoN[f],FtoN[f]); 

A_f.append(al)/ 

partner[al]= al;a_in_P[al]= e; 

}else{ 
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al= N.new_edge(FtoN[f],FtoN[g]); 
a2= N.new_edge(FtoN[g],FtoN[f]) ; 

A_f.append(al);A_f.append(a2 ) ; 
partner[al]= a2;partner[a2]= al; 
a_in_P[al]= e;a_in_P[a2]= e; 
edge e_loop; 
for(e_loop= e; 

P. face__of (REV (e_loop) ) ==g&& !marked [e_loop] ; 
e_loop= SUCC(e_loop)) 

marked[e_loop]= marked[REV(e_loop)]= true; 
for(e_loop= PRED(e); 

P.face_of(REV(e_loop))==g&&Imarked[e_loop]; 
e_loop= PRED(e_loop)) 

marked[e_loop]= marked[REV(e_loop)]= true; 

} 

} 


cap.init(N,0);cost.init(N,0);int z_s= 0;edge a; 

1.init(N,0); 

forall(a,A_s_f)cap[a]= 4~P.size(NtoF[target(a)]); 

forall(a,A_s_v)cap[a]= 4-outdeg(NtoV[target(a)])/ 

forall(a,A_v)cap[a]= 3; 

forall(a,A_v_cage){cap[a]= 3;1[a]= 1;} 

forall(a,A_f){ 

face g= NtoF[target(a)];face f= NtoF[source(a)]; 
cap[a]= IS_CAGE(g)?0:MAX_BENDS_PER_EDGE; 
cost[a]= IS_CAGE(f)?0:1; 

} 

forall(a,A_f_t){ 
face f= NtoF[source(a)]; 
if(IS_OUTER(f))cap[a]= P.size(f)+4; 
else cap[a]= P.size(f)~4; 

} 

forall_out_edges(a,s)z_s+= cap[a]; 
forall_in_edges(a,t)z+= cap[a]; 

A_v.cone(A_v_cage); 

if(z!=z s)error handler(1,ERR_INVALID_NETWORK); 


//cout<<’’network constructed”<<endl; 


edge_array<int>flow(N); 
node_array<int>supply(N,0); 
supply[s]= z;supply[t]= -z; 

bool feasible= MIN_COST_FLOW(N,1,cap,cost,supply,flow); 
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if ( ! feasible) error_handler (1, ERR_NO_FEASIBLE_FLOW) ; 
//cout<<”min-cost flow computed"<<endl; 


forall(a,A_v){ 

node v= NtoV[source(a)];face f= NtoF[target(a)]; 
edge e_in;bool flag= false; 
forall_in_edges(e_in,v) 
if(P.face_of(e_in)==f&&!flag){ 

P.set_a(e_in,(flow[a]+1)*90); 
flag= true; 

} 

} 


marked.init(N,false);int no_of_bends= 0; 

forall(a,A_f)if(!marked[a] ) { 

edge a_rev= partner[a]; 

marked[a]= true;marked[a_rev]= true; 

e= a_in_P[a]; 

bool bridge= (a==a_rev); 

P.set_s(e,flow[a],flow[a_rev],bridge); 
P.set_rev_s (REV (e) , flow[a]flow[a_rev] , bridge) ; 
no_of_bends+= (flow[a]+flow[a_rev]); 

} 


marked.init(P,false); 
list<edge>all_edges= P.all_edges(); 
forall(e, all_edges)if(!marked[e]) { 
marked[e]= true;marked[REV(e)]= true; 
edge e_orig= P.get_orig(e); 
while(P.get_s(e)!=EPS){ 
edge e_split= P.split_bend_edge(e); 
if(e_orig){ 

b_nodes[e_orig].push(source(e_split)); 

b_nodes[G.reversal (e_orig)] .append(source(e_split)) ; 

} 

} 

if(e_orig){ 

if(b_nodes_first[e_orig]){ 

b_nodes[e_orig].push(b_nodes_first[e_orig]); 
b_nodes[G.reversal(e_orig)].append(b_nodes_first[e_orig]) 
} 

if(b_nodes_last[e_orig] ) { 

b_nodes[e^orig].append(b_nodes_last[e_orig]); 

b^nodes[G.reversal(e_orig)].push(b_nodes_last[e_orig]); 

} 

} 


193 


} 


//cout<<"orthogonal representation constructed"«endl; 

//if (check) if ( ! P. check {) ) error_handler {1, ERR_NO_ORTHO_REP) 
//cout<<"no. of bends in 4-graph: "<<no_of_bends«endl; 


list<face>all faces; 


forall_faces{f,P)if(IS_OUTER(f))all_faces.push(f) ; 
else { 

if(IS_CAGE{f)) { 

int pred_angle= P.get_a(PRED(P.first_face_edge(f))) ; 

forall_face_edges(e,f){ 

int act_angle= P.get_a(e); 

if(pred_angle==90&&act_angle==90) { 

edge e_split= P.split_map_edge(e) ; 

P. set_a (e, 180) ; P. set_a (REV (e_split) , 180) ; 

P. set_type (source (e_split) , dissection) ; 

} 

pred_angle= P.get_a(e); 

} 

} 

all_faces.append(f); 

} 


forall(f,all_faces){ 

stack<edge>S/ 
int size= 0; 

forall_face_edges(e,f)size+= abs(2-P.get_a(e)/90) ; 

e= P. succ_corner_edge (P. first_face__edge (f) ) ;int state= 0; 

while(size>4){ 

if (state==0 &&!S.empty()){ 

e= P.succ_corner_edge(S.top() ) ; 

state= 1; 

} 

int angle= P.get_a(e); 
switch(angle){ 

case 90:if(state==2)state= 3; 
if(state==l)state= 2; 
break; 
case 270: 

case 360 : if(state==2&&IS_OUTER(f) )state= 4/ 
else{ 

S.push(e);state= 1; 
if(angle==360)S.push(e); 

} 

break; 

} 

if(state==3){ 
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edge el= S.pop(); 

edge e2= P.succ_corner_edge(el) ; 

edge e3= P.succ_corner_edge(e2); 

edge e4= P.succ_corner_edge(e3);int a4= P.get_a(e4); 
edge e5= P . split_inap_edge (e4 ) ; 

P.set_type(source(e5),dissection); 

P.set_a(el,P.get_a(el)-90) ; 

P.set_a(e5,a4);P.set_a(REV(e5),180); 

P.set_a(e4,90); 

edge e6= P.new_edge(P.face_cycle_succ(el) , e5) ; 

P.init_maps {e6, 90, EPS,P.gethinner(el),false) ; 

P.init_maps(REV(e6),90,EPS,true, false) ; 

P . set_inner(e2,true);P.set_inner(e3,true);P.set_inner(e4,true); 
size-= 2;state= 0;e= e6; 


} 

if(state==4){ 

edge el= S.pop(); 

edge e2= P.succ_corner_edge(el) ; 

edge e3= P.succ_corner_edge(e2) ; 

edge e4= P.new_edge (P. face_cycle_succ (el) , P. face_cycle__succ (e3) ) 
P . init_maps (e4 , P. get_a (e3) -90, " 1*', false, false) ; 

P.init_maps(REV(e4),90,"0”,true,false); 
edge e5= P.split_bend_edge(e4) ; 

P.set_a(el,P.get_a(el)-90) ; 

P.set_inner(e2, true) ; 

P.init_maps(e3,90,EPS,true, false) ; 

P.set_type(source (e5),dissection); 

S.push(el)/ 
size-= 2;state= 0; 


} 

e= P.succ_corner_edge(e) ; 

} 


} 

//if(check&&!P.check 0)error handler(1, ERR NO ORTHO REP); 


P.init_rest(); 

P.assign_directions(P.first_edge() , north) ; 


#ifdef CPLEX 
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int n= 0;int m= 0; 

node_array<int>v_num(P);edge_array<int>e_num(P,-1); 
forall_nodes(V,P)v_num[v]= n++; 

forall_edges(e,P)if(P.get_dir(e)==north||P.get_dir(e)==west) 

e_num[e]= m+ +; 

int basic_rows= 2*m; 

int rows= basic_rows; 

int cols= 2*n; 

int basic_nonzeroes= 2*basic_rows; 
int nonzeroes= basic_nonzeroes; 
double*obj= new double[cols]; 
double*rhs= new double[rows]; 
char*sense= new char[rows]; 
int*matbeg= new int[cols]; 
int*matcnt= new int[cols]; 
int'*^matind= new int [nonzeroes] ; 
double*matval= new double[nonzeroes]; 
double*lb= new double[cols]; 
double*ub= new double[cols]; 

CPXENVptr env= NULL; 

CPXLPptr lp= NULL; 

int status; 


for(int j= 0;j<cols;j++){ 
obj[j]= 0.0; 

matbeg[j]= 0;matcnt[j]= 0; 
lb[j]= 0.0;ub[j]= INFBOUND; 

} 

for(int k= 0;k<basic_nonzeroes;k++){ 
matind[k]= 0;inatval [k] = 0.0; 

} 


#define X(v) (v_num[v]) 

#define Y(v) (n + v_num[v]) 

forall_edges(e,P){ 

node s= source(e),t= target(e); 

switch(P.get_dir(e)){ 

case north: 

obj [ v_nuin [ s ] ] -= 1.0; 

obj [ v_nuin [ t ] ] += 1.0; 

break; 

case west: 

obj[v_num[s]+n]-= 1.0; 
obj [ v_nuin [ t ]+n]+= 1.0; 
break; 

default:break; 

} 

} 
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int nonzero_cnt= 0; 
forall_nodes(v, P) { 
int act_col= X(v); 
matbeg[act_col]= nonzero_cnt; 
forall_inout_edges(e,v){ 

bool out_edge= (v==source(e));int act_row 

switch(P.get_dir(e)){ 

case north: 

act_row= 2*e_num[e]; 

matcnt[act_col]++; 

if(out_edge)matval[nonzero_cnt]= 1.0; 

else matval[nonzero_cnt]= -1.0; 

matind[nonzero_cnt]= act_row; 

sense[act_row]= 'L*;rhs[act_row]= -1.0; 

++nonzero_cnt; 

break; 

case west: 

act_row= 2*e_nijm [e]+1; 
matcnt[act_col]++; 

if(out_edge)matval[nonzero_cnt]= 1.0; 

else matval[nonzero_cnt]= -1.0; 

matind[nonzero_cnt]= act_row; 

sense[act_row]= ’E’;rhs[act_row]= 0.0; 

++nonzero_cnt; 

break; 

default -.break; 

} 

} 

} 

forall_nodes(v,P){ 
int act_col= Y(v); 
matbeg[act_col]= nonzero^cnt; 
forall_inout_edges(e,v){ 

bool out_edge= (v==source(e));int act_row 
switch(P.get_dir(e)){ 
case west: 

act_row= 2*e_num[e]; 
matcnt[act_col]++; 

if(out_edge)matval[nonzero_cnt]= 1.0; 

else matval[nonzero_cnt]= -1.0; 

matind[nonzero_cnt]= act_row; 

sense[act_row]= *L*;rhs[act_row]= -1.0; 

++nonzero_cnt; 

break; 

case north: 

act_row= 2*e_num[e]+1; 
matcnt[act_col]++; 

if(out_edge)matval[nonzero_cnt]= 1.0; 

else matval[nonzero_cnt]= -1.0; 

matind[nonzero_cnt]= act_row; 

sense[act_row]= *E*;rhs[act_row]= 0.0; 

++nonzero_cnt; 

break; 

default:break; 

} 

} 
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} 

env= CPXopenCPLEX (ScStatus) ; 

if(env==NULL)error_handler(1,ERR_OPEN_CPLEX_ENV); 

lp= CPXloadlp(env,”find_coords",cols,basic_rows,CPX_MIN,obj,rhs, 

sense,matbeg,matcnt,matind, 

matval,lb,ub,NULL,cols,rows,nonzeroes) ; 

if(lp==NULL)error_handler(1,ERR_LOAD_LP); 


int solstat; 
double objval; 

double*sol= new double[cols]; 
double*pi= new double[rows]; 
double*slack= new double[rows]; 
double*dj= new double[cols]; 

status= CPXoptimize(env,Ip); 
if(status){ 

CPXcloseCPLEX(&env); 
error_handler(1,ERR_CPLEX_OPT) ; 
} 


status= CPXsolution(env,Ip,&solstat,&objval,sol,pi,slack,dj); 
if(status)error_handler(1,ERR_CPLEX_GET_SOLUTION); 
int max_x= 0,max_y= 0; 

forall_nodes(v, P)if(P.get_type(v)==real) { 

max_x= Max(max_x,(x_pos[P.get_orig(v)]= (int)sol[X(v)])); 
max_y= Max (itiax_y, (y_pos [P. get_orig (v) ] = (int) sol [Y (v) ] ) ) ; 

} 

if(env)status= CPXfreeprob(env,&lp); 

if(status)error_handler(1,ERR_CPLEX_FREE); 

status= CPXcloseCPLEX(&env); 

if(status)error handler(1,ERR_CPLEX_CLOSE); 


#else 


graph N_h,N_v; 

face_map<node>FtoN_h(P),FtoN_v(P) ; 
edge_inap<edge>EtoA (P) ; 


node s_h= N_h.new_node(),t_h= N_h.new^node(),s_v= N_v.new_node() , 
t_v= N_v.new_node(); 

forall_faces(f,P){ 
if(!IS_0UTER(f)){ 

FtoN h[f]= N h.new node() ; 
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FtoN_v[f]= N_v.new_node(); 

} 

} 


forall_faces(f,P)forall_face_edges(e, f) { 
face g= P.face_of(REV(e)); 

bool out_f= IS_OUTER(f),out_g= IS_OUTER(g); 
switch(P.get_dir(e)){ 
case north: 

if (out_f&& ! out_g) EtoA[e] = N_h. new_edge (FtoN_h [g] , t_h) ; 
if (out_g&& ! out_f) EtoA [e] = N_h. new_edge (s_h, FtoN_h [ f ] ) ; 
if (out_f &&out_g) EtoA [e] = N_h.new_edge (s_h, t_h) ; 
if ( 1 out_f&& ! out_g) EtoA[e] = 

N_h. new_edge (FtoN_h [g] , FtoN_h [ f ] ) ; 

break; 

case west: 

if (out_f && ! out_g) EtoA [e] = N_v. new_edge (FtoN_v [g] , t_v) ; 
if(out_g&&!out_f)EtoA[e]= N_v.new_edge(s_v,FtoN_v[f]); 
if(out_f&&out_g)EtoA[e]= N_v.new_edge(s_v,t_v); 
if(!out_f&&!out_g)EtoA[e]= 

N_v.new_edge(FtoN_v[g],FtoN_v[f]) ; 
break; 

default -.break; 

} 

} 

edge e_h= N_h. new_edge (s_h, t_h) ; 
edge e_v= N_v.new_edge(s_v,t_v) ; 


edge_array<int>cap_h(N_h,INFINITY),cap_v(N_v, INFINITY); 
edge_array<int>cost_h(N_h,1),cost_v(N_v, 1) ; 
edge_array<int>l_h(N_h,1),l_v(N_v, 1) ; 
edge_array<int>flow_h(N_h,0) , flow_v(N_v,0); 
node_array<int>b_h(N_h,0),b_v(N_v, 0) ; 
forall_nodes(v,N_h) 

b_h[s_h]+= abs(outdeg(v)-indeg(v)); 
b_h[s_h]= b_h[s_h]/2+1; 
b_h[t_h]= -b_h[s_h]; 
forall_nodes(v,N_v) 

b_v[s_v]+= abs(outdeg(v)-indeg(v)) ; 
b_v[s_v]= b_v[s_v]/2+l; 
b_v[t_v]= -b_v[s_v]; 
cost_h[e_h]= 0;cost_v[e_v]= 0; 
l_h[e_h]= 0;l_v[e_v]= 0; 


feasible= MIN_COST_FLOW(N_h,l_h,cap_h, cost_h, b_h,flow_h) 
if(!feasible)error_handler(l,ERR_NO_FEASIBLE_FLOW); 
feasible= MIN_COST_FLOW (N_v, l_v, cap_v, cost_v,b_v, flow_v) 
if(!feasible)error_handler(1,ERR_NO_FEASIBLE_FLOW); 


199 


forall_edges(e,P){ 
switch(P.get_dir(e)){ 
case north: 

P.set_length(e,flow_h[EtoA[e]]); 
P.set_length(REV(e),flow_h[EtoA[e]]); 
break; 
case west: 

P.set_length(e,flow_v[EtoA[e]]); 

P.set_length(REV(e),flow_v[EtoA[e]]); 
break; 

default:break; 

} 

} 


//for(v= P.choose_node();P.get_type(v)!=real;v= P.succ_node(v)) 

forall_nodes(v, P) 

if (P.get_type(v) == real) break; 

x_pos.init(G);y_pos.init(G);node_array<bool>seen(P, false); 
P.determine_position(v,0,0,seen); 

P.norm_positions(); 
int max_x= 0,max___y= 0; 

forall_nodes (v, P) if (P . get_type (v) ==real) { 

int X = P.get_x(v); 

int y = P.get_y(v); 

x_pos[P.get_orig(v)]=x; 

y_pos[P.get_orig(v)]=y; 

max_x= Max(max_x,x); 

max_y= Max(max_y,y); 

} 


//cout<<"coordinates computed by network method"<<endl; 


#endif 


forall_nodes(v,P)if(P.get_type(v)==big){ 
int x_big,y_big; 
list<int>lx,ly; 

face f_cage= P.face_of(corr_cage_edge[v]) ; 

forall_face_edges(e,f_cage){ 

if(P.get_type(source(e))!=dissection){ 

int x= P.get_x(source(e)),y= P.get_y(source(e)); 

lx.append(x);ly.append(y); 

} 

} 

lx.sort();ly.sort();list_item it; 
x_big= (lx.head()+lx.tail())/2; 
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y_big= {ly.head{)+ly.tail())/2; 
forall_items(it,lx) 

if{lx[it]!=lx.head{)&&lx[it]!=lx.tail{) 

&&lx[it]==lx[lx.cyclic_succ(it)]) x_big= lx[it]; 

forall^items(it,ly) 

if(ly[it]!=ly.head()&&ly[it]!=ly.tail() 

&&ly[it]==ly[ly.cyclic_succ(it)]) y_big= ly[it] 

x_pos[P.get_orig(v)]= x_big; 
y_pos[P.get_orig(v)]= y_big; 

} 

x_bends.init(G);y_bends.init(G); 

forall_edges(e, G) 

forall(V,b_nodes[e]){ 

x_bends[e].append(P.get_x(v)); 

y_bends[e].append(P.get_y(v)); 

} 

//cout<<"cages resolved"<<endl<<endl; 


return Max(max_x,max_y); 

} 
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'k’k'k'k'k'k-k-k 

+ 

+ LEDA 3.5.1 
+ 

+ _spring.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

■^(‘k'k-k-k'k-k-k'k-k-k-k'k'k'k'k'k'k'k'k-k-k'k-k-k-k-k'k-k-k-fe-k'k'k'k'k-ir'k'k'k'k-k-k’k'k-k-k'k-k-k-k'k-fe'k'k'k-k-k-k-k-k'k'k-k'k'k'k-k-k'k-k'k 
•k -k 'k -k 'k -k -k J 

#include <LEDA/graph_alg.h> 

#include <math.h> 

#include <LEDA/array2.h> 


#define FREPULSE(d) ((k2 > d) ? kk/d : 0) 

static float log_2(int x) 

{ float 1 = 0; 
while (x) 

{ 1 ++; 

X »= 1; 

} 

return 1/2; 


void SPRING_EMBEDDING(const graph& G, node_array<double>& xpos, 

node_array<double>& ypos, 
double xleft, double xright, 
double ybottom, double ytop, 
int iterations) 

{ list<node> L; 

SPRING_EMBEDDING(G,L,xpos,ypos,xleft,xright,ybottom,ytop,iterations); 

} 


void SPRING_EMBEDDING(const graph& G, const list<node>& fixed_nodes, 

node_array<double>& xpos, 
node_array<double>& ypos, 
double xleft, double xright, 
double ybottom, double ytop, 
int iterations) 

{ 


if (xleft >= xright |1 ybottom >= ytop) 

error_handler(1,"SPRING_EMBDDING: illegal bounds."); 

node_array<list_item> lit(G); 
node_array<bool> fixed(G,false) ; 
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node u,v; 
edge e; 

forall(v, fixed_nodes) fixed[v] = true; 
int c_f = 1; 

double width = xright - xleft; 
double height = ytop - ybottom; 

double tx_null = width/50; 
double ty_null = height/50; 
double tx = tx_null; 
double ty = ty_null; 

double k = sqrt(width*height / G.number_of_nodes()) / 2; 
double k2 = 2*k; 
double kk = k*k; 

int ki = int(k); 

if (ki == 0) ki = 1; 

//build matrix of node lists 

int xA = int(width / ki + 1); 
int yA = int(height / ki + 1); 

array2<list<node> > A (*“1, xA,-1, yA) ; 

forall_nodes(v,G) 

{ int i = int((xpos[v] - xleft) / ki); 
int j = int((ypos[v] - ybottom) / ki); 

if (i >= xA II i < 0) error_handler(1,"spring: node out of range") 
if (j >= yA II j < 0) error_handler(1,"spring: node out of range") 
lit[v] = A(i,j).push(v); 

} 


while (c_f < iterations) 

{ 

node_array<double> xdisp(G,0); 
node_array<double> ydisp(G,0); 

// repulsive forces 

forall_nodes(v,G) 

{ int i = int((xpos[v] - xleft) / ki); 
int j = int((ypos[v] - ybottom) / ki); 

double XV = xpos[v]; 
double yv = ypos[v]; 

for(int m = -1; m <= 1; m++) 
for(int n = -1; n <= 1; n++) 
forall(u,A(i+m,j+n)) 

{ if (u == v) continued- 

double xdist = XV - xpos[u]; 
double ydist = yv - ypos[u]; 

double dist = sqrt(xdist * xdist + ydist * ydist); 
if (dist < le-3) dist = le-3; 
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xdisp[v] += FREPULSE(dist) * xdist / dist; 
ydisp[v] += FREPULSE(dist) * ydist / dist; 

} 

xdisp[v] *= (double(rand_int(750,1250))/lOOO. 0) 
ydisp[v] *= (double(rand_int(750,1250))/1000.0) 

} 


// attractive forces 

forall_edges(e,G) 

{ node u = G.source(e); 
node V = G.target(e); 
double xdist=xpos[v]-xpos[u]/ 
double ydist=ypos[v]-ypos[u]; 
double dist=sqrt(xdist*xdist+ydist*ydist) ; 

float f = (G.degree(u)+G.degree(v))/6.0; 

dist /= f; 

xdisp[v]-=xdist*dist/k; 
ydisp[v]-=ydist*dist/k; 
xdisp[u]+=xdist^dist/k; 
ydisp[u]+=ydist*dist/k; 

} 


// preventions 

forall_nodes(v,G) 

{ 

if (fixed[v]) continue; 

int iO = int((xpos[v] - xleft)/ki); 
int jO = int((ypos[v] - ybottom)/ki); 

double xd= xdisp[v]; 
double yd= ydisp[v]; 
double dist = sqrt(xd*xd+yd*yd) ; 

if (dist < 1) dist = 1; 

xd = tx*xd/dist; 
yd = ty*yd/dist; 

double xp = xpos[v] + xd; 
double yp = ypos[v] + yd; 

int i = iO; 
int j = jO; 

if (xp > xleft && xp < xright) 

{ xpos[v] = xp; 

i = int((xp “ xleft)/ki); 

) 

if (yp > ybottom && yp < ytop) 

{ ypos[v] = yp; 

j = int((yp - ybottom)/ki); 
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) 


if (i != iO I I j != jO) 

{ if (lit[v] == nil) error_handler{1,"delete nil item") 
A(iO, j 0).del_item(lit[v]) ; 
lit[v] = A(i,j).push{v); 

} 


} 


tx = tx_null 
ty = ty_null 


/ log_2(c_f); 
/ log_2(c_f); 


c_f++; 

} 


} 
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^■^c-k-k-k-kir'^'^(-k'^-k-k-kir-k-k-k-k-k****-k-k-k*-k*-k-k*-k-k*ir*-if-kic*-k*-k-k-k-k-k-k-k-k-k-k-ic-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k 
★ •A-****** 

+ 

+ LEDA 3*5.1 
+ 

+ _sugiyama.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, FRG 
+ (fax +49 681 31104). 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 

+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 

•^('k-k'k'k-k'k'k'k'k'k'k'k'k-k'k-k'k'k-k'k'k'k'k-k'k-k-k-k'k'^f-k'k'k-k-k-k-k-k'k'k-k-k’k'if'k-k-k-k'k-k'k'k'k'k-k'k-k'k'k'k-k-k'k'k'k'k'k'k'k'k-k 
•k -k -k -k -k -k -k j 

I j - 

// SUGIYAMA EMBEDDING 

// 

// D. Ambras (1996/97) 

// - 


#include 

#include 

#include 

#include 

#include 

#include 


<stdio.h> 
<LEDA/array.h> 
<LEDA/p_queue.h> 
<LEDA/stream.h> 
<LEDA/graph.h> 
<LEDA/graph_alg,h> 


#ifdef _DRAW_SUGI 
#include <LEDA/panel.h> 
#endif 


typedef int bitarray; 


#define SORT_LOWER 1 
#define SORT_UPPER 2 
#define SORT_BOTH 3 
#define PERMUTE 4 
#define IMPROVE 8 


#define HIGH 1 
#define LOW 0 
#define UP 1 
#define DOWN 0 


#ifdef 

window 

#endif 

double 

int 

int 

int 

bool 


DRAW_SUGI 

W; 

aspect__rat io=3; 
ps_count=l; 

perm_max=10, sort__max=6; 
max_level; 
do_debug=t rue; 
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void write_to_ps(const graph &G, array<list<node> > &Level, 
node_array<int> 

&x_coord, node_array<int> &nlabel, int width, int 

height, 

char ^headline, file_ostreain &out) 

{ 

int x_off, y_off, x_gap, y_gap, radius, i; 
x_off = y_off= x_gap= 20; 
y_gap = int(x_gap * aspect_ratio + 0.5); 
radius = 3; 

node V, s, t; 
edge e; 

out « ”%!PS\n”; 

out « "%%Creator: SUGIYAMA_EMBEDDING vl.0\n"; 

out « ”%%Pages: l\n"; 

out << ”%%PageOrder: Ascend\n"; 

out << ”%%BoundingBox: " << 0 « ” ” « 0 << ” ”; 

out « x_gap*width + 2*x_off << ” " « y_gap*height + 2*y_off << ”\n” 
out << "%%EndComments\n\n%%Page: 1 l\n\n"; 

out << ”% ” « headline « endl << endl; 


out « ”% drawing the nodes\n\n"; 
i=-l; 

while ((++i) <= max_level) 
forall(v. Level[i]) if (nlabel[v] != 0) 

{ out << "newpath ”; 

out « x_gap *x_coord[v] +x_off « " " « y_gap *i +y_off; 
out « " ” << radius « ” 0 360 arc fill\n”; 

} 

out << ”\n% drawing the edges\n\nnewpath\n”; 
i-0; 

while ( (++i) <= max_level) // drawing the 

edges 

forall(v. Level[i-1]) 
forall_out_edges(e, v) 

{ s= G.source(e); t= G.target(e); 

out << x_gap *x_coord[s] +x_off « ” " << y_gap*(i-1) +y_off 
<< " moveto ”; 

out << x_gap *x_coord[t] +x_off « " ” << y_gap*i +y_off 
<< " linetoXn”; 

} 

out << ”stroke\nshowpage\n\n%%EOF” << endl « flush; 

} 


207 


#ifdef _DRAW_SUGI 

void user_exit(void) 

{ 

cout << "Aborted." « endl << flush; 
exit(1); 

} 


bool draw_hierarchy(const graph &G, array<list<node> > &Level, 

node_array<int> &x_coord, node_array<int> &nlabel, 
char ^headline, const bool &draw_virtual_nodes) 

{ 

int i, x_min=100, x_max=“100, xv, b=NO_BUTTON; 

int width=0, height; 
double W_size; 

color bkg_color=white, node_color, edge_color=blue; 
color rnode_color=green, vnode_color=white; 
node V, w; 

edge e; 

char filename[200]; 


forall_nodes(v, G) 

{ i=x_coord[v] ; 

if (x_min > i) x_min= i; 
if (x max < i) x max= i; 


width= x_max -x_min +1; 
height=max__level + 1; 


W_size=(width > height*aspect_ratio ? width: height*aspect_ratio); 
W.init(~l, W_size, “1); 

W.clear(bkg_color); W.set_frame_label(headline); 


#ifdef _DEBUG_SUGI 
if (do_debug) 

{ W.message("writing graph to disk 

sprintf(filename, "figure%d.ps", ps_count); 
file_ostream ps_out(filename); 
write_to_ps(G, Level, x_coord, nlabel, width, 
height, filename, ps_out); 

ps_count++; 

} 

#elif _DEBUG2_SUGI 
if (do_debug) 

{ W.message("writing graph to disk ..."); 
sprintf(filename, "figure%d.ps", ps_count); 
file_ostream ps_out(filename); 
write_to_ps(G, Level, x_coord, nlabel, width, 
height, filename, ps_out); 

ps_count++; 

} 


208 


#endif 


i=0; forall(v, Level[i]) 

W. draw_int_node {x_coord [v]-x_inin, i*aspect__ratio, nlabel[v], 
rnode__color) ; 

do 

{ if (i == max_level) //no more levels 

{ W.message( 

"press left mouse button to proceed, other button to abort"); 
while (b == NO_BUTTON) b= W.get_button(); 

W.del_messages(); 
if (b==MOUSE_BUTTON(1)) 

{ W.message("continue calculations, please wait 
return true; 

} 

return false; 

} 


forall{v. Level[i]) // drawing the 

edges 

{ xv=x_coord[v]-x_min; 
forall_out_edges(e, v) 

if {draw_virtual_nodes || {nlabel[v] > 0 && nlabel[G.target(e)] > 

0 ) ) 


} 


W.draw_edge(xv, i*aspect_ratio, x_coord[G.target(e)] “X_min, 
(i+l)*aspect_ratio, edge_color); 

else 

W.draw_segment (xv, i*aspect_ratio, x_coord[G.target(e)]-x_min, 
(i+l)*aspect_ratio, edge_color); 


forall(v. Level[i+l]) . // drawing the 

nodes 

{ if (nlabel[v]==0) node_color=vnode_color; 
else node_color=rnode_color; 
if (draw_virtual_nodes || nlabel[v] > 0) 

W.draw_int_node (x_coord[v] -x_min, (i+l) *aspect_ratio, 
nlabel [v] , node__color) ; 

} 

i++; 

} while (0<1); 

} 


#endif 


bool slide_node(const graph &G, const node &v, const int &priority, 

const int &best_pos, node_array<int> &x_coord, node_array<int> 

&x_prio, 

array<bool> &x_set, array<node> &x_owner) 

{ 

int cur_pos=x_coord [v] , N=G.number_of_nodes () ; 
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while (cur_pos < best_pos) 

{ if (cur_pos > N-1) return false; 

if (x_set[cur_pos+l]) 

{ if (x_prio[ x_owner[cur_pos+l] ] > priority) 
return false; 


if ( !slide_node(G, x_owner[cur_pos+l] , 

priority, cur_pos+2, 
x_coord, x_prio, 

X set, X owner)) return false; 


others not 

} 

x_set[cur_pos++]=false; 
x_set[cur_pos]=true; 
x^owner[cur^pos]=v; 
x_coord[v]=cur_pos; 

} 


// owner was move-able, but 


// move node one step 


while (cur_pos > best_pos) 

{ if (cur_pos < -N+1) return false; 

if (x_set[cur_pos-l]) 

{ if (x_prio[ x_owner[cur_pos-l] ] >= priority) 
return false; 

if { 1slide_node(G, x_owner[cur_pos-l] , 

priority, cur_pos-2, 

x_coord, x_prio, 

x_set, x_owner)) return false; 

} 

x_set[cur_pos—]=false; 

x_set[cur_pos]=true; 

x^owner[cur_pos]=v; 

x_coord[v]=cur_pos/ 


return true; // successful move 

} 


bool two_level(const graph &G, array<list<node> > &Level, const 
&upper, 

node_array<int> &x_coord, node_array<int> &x_prio, 
const bitarray &what_to_do) 

{ 


bool changes=false; 

int lower=upper+l, N=G. nuitiber_of_nodes {) ; 


int 
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int 

node 

node 

edge 

double 


X, count, pcount, hprio_count; 
nu, nl; 
u, v; 
e; 

bary, old__bary, hprio_bary; 


list<node> equal_bary/ 

p_queue<double, node> Su, SI; 
pq_item pit; 


array<bool> 

array<node> 

node_array<int> 


x_set(-N, N); 
x_owner(-N, N); 
p_bary(G); 


if (what_to_do & SORT_UPPER) 

{ forall(nu, Level[upper]) 

{ bary=0; 

hprio_count=0; 
hprio_bary=0/ 

forall_out_edqes(e, nu) 

{ x=x_coord[ G.target(e) ]; bary+=x; 

if (x_prio[ G.target(e) ] == N) // a long edge 

{ hprio_count++; 
hprio_bary+=x; 

) 

} 


if (count=G.outdeg(nu)) bary =bary/count; 

else bary=x_coord[nu]; // no in- edges 

if (what_to_do & IMPROVE) 

{ Su. insert (-x__prio [nu] , nu) ; // sort by priority 

p_bary[nu]=(int)(bary+0.5); 
if (hprio^count) 

p_bary[nu]=(int)(hprio_bary/ hprio_count +0.5); 

// we ignore short edges 

} 

else Su.insert(bary, nu); 


if (what_to_do & IMPROVE) 

{ for(x=-N; x<= N; x++) x_set[x]=false; 
forall(v. Level[upper]) 

{ x_owner[ x_coord[v]]=v; 
x_set[ x_coord[v]]=true; 

} 

} 


count=0; old_bary=-l; pcount=-2; 

while(Su.size()) 

{ pit=Su.find_min(); 
v=Su.inf(pit); 
bary=Su.prio(pit) ; 

Su.del_item(pit); 
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if (what_to_do & IMPROVE) 
x_coordinates 

slide node(G, v, 


// set real 


x_prio[v] , p_bary[v], 
x_coord, x_prio, 

X set, X owner); 


else if (x coord[v] != count) 


// set consecutive 


x__coordinates 

{ changes=true; 


X coord[v] = count; 


if (what to do & PERMUTE) 


//we shall permute 


if (bary“old_bary) 

{ equal_bary.append(u); 
pcount=count; 

} 

else 

{ if (pcount == count-1) equal_bary.append(u) ; 
while (!equal_bary.empty()) 

{ changes=true; 

x_coord[equal_bary.pop()]= pcount--; 

} 

} 

count++; 

old_bary=bary; 

u=v; 

} 

if (pcount == count-1) equal_bary.append(u); 

while (!equal_bary.empty 0) // flush equal_bary 

{ changes=true; 

x_coord[equal_bary.pop()]= pcount—; 

) 


if (what_to_do & SORT__LOWER) 

{ forall(nl, Level[lower]) 

{ bary=0; 

hprio_count=0; 
hprio_bary=0; 

forall_in_edges(e, nl) 

{ x=x_coord[ G.source(e) ]; bary+=x; 

if (x_prio[ G.source(e) ] == N) // a long edge 

{ hprio_count++; 
hprio_bary+=x; 

} 

} 

if (count=G.indeg(nl)) bary =bary/count; 
else bary=x_coord[nl] ; 

if (what_to_do & IMPROVE) // set real 

x_coordinates 

{ SI. insert (-x__prio [nl] , nl); 
p_bary[nl]=(int)(bary +0.5); 
if (hprio_count) 
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p_bary[nl]=(int)(hprio_bary/ hprio_count +0.5); 

// ignore short edges 

} 

else SI.insert(bary, nl); 


if (what_to_do & IMPROVE) 

{ for(x=-N; x<= N; x++) x_set[x]=false; 
forall(v, Level[lower]) 

{ x_owner[ x_coord[v]]= v; 
x_set[ x_coord[v]]= true; 

} 

} 


count=0; old_bary=-l; pcount=-2; 

while(SI.size()) 

{ pit=Sl.find_min(); 
v=Sl.inf(pit); 
bary=Sl.prio(pit); 

SI.del_item(pit); 

if (what_to_do & IMPROVE) 
slide_node(G, v, 

x_prio[v], p_bary[v], 
x_coord, x_prio, 
x_set, x_owner); 

else if (x_coord[v] != count) 

{ changes=true; 

x_coord[v] = count; 

} 

if (what_to_do & PERMUTE) 
if (bary==old_bary) 

{ equal_bary.append(u); 
pcount=count; 

} 

else 

{ if (pcount == count-1) equal_bary.append(u); 
while (!equal_bary.empty 0) 

{ changes=true; 

x_coord[equal_bary.pop()]=pcount — ; 

} 

) 

count++; 

old_bary=bary; 

u=v; 

) 

if (pcount == count-1) equal_bary.append(u) ; 

while (!equal_bary.empty 0) // flush equal_bary 

{ changes=true; 

x_coord[equal_bary.pop()]=pcount — ; 

} 


#ifdef DEBUG2 SUGI 
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char 


headline [200] , dir [10], text__p[10], text_i[10]; 


if ({ upper > 10) && do_debug) 

{ if (what_to_do & SORT_LOWER) sprintf(dir, "[down]”); 
else sprintf(dir, ”[up]”); 

// ( supposed SORT_BOTH is not selected ) 

if (what_to_do & PERMUTE) sprintf(text_p, ” + permute"); 
else sprintf(text_p, ” ”); 

if (what_to_do & IMPROVE) sprintf (text_i, " + improve”); 
else sprintf(text_i, ” ”); 


sprintf(headline, ”after two_level for u-level %d, sort %swards%s%s”, 
upper, dir, text_p, text_i); 

if (!draw_hierarchy (G, Level, x_coord, x_coord, headline, true)) 
user_exit(); 

} 

#endif 


return changes; 

} 


list<edge> sort_edges(const graph &G, array<list<node> > &Level, 

const int &column, const int &height, 
node_array<int> &x_coord, list<edge> &edges) 

{ 

int i=0/ 

node v; 

edge e; 

list<edge> result; 

array< list<edge> > nlist( Level[column] .size{) ); 

forall(e, edges) 

{ if (height == LOW) v=G.target(e) ; 
else v=G.source(e); 
nlist [ x_coord[v] ].append(e); 


while (i < Level[column].size() ) 

result.cone(nlist[i++]); 

return result; 

} 


static bool cross (const graph &G, node_array<int> &x_coord, 

edge_array<int> &current_pos, const edge &e, const 

edge &f) 

{ 

int xul, xu2, xll, xl2; 

xul=current_pos[e]; 
xu2=current_pos[f]; 
xll=x_coord[ G.target(e) ]; 
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xl2=x_coord[ G.target(f) ]; 

return ( xul < xu2 && xll > xl2) || ( xul > xu2 && xll < xl2 ); 


} 


int 

{ 


nuinber_of_crossings (const graph &G, array<list<node> > &Level, 

const int &upper, node_array<int> &x_coord) 


int 

node 

edge 

list<edge> 
list item 


lower=upper+l, result=0, i; 
vu; 

e, f, q, h; 
edges, work_list; 
lit ; 


forall(vu, Level[upper]) 
forall_adj_edges(e, vu) edges.append(e); 

edges= sort_edges(G, Level, lower, LOW , x_coord, edges); 
edges= sort_edges(G, Level, upper, HIGH, x_coord, edges); 


array<edge> current_order(edges.size() ); 

edge_array<int> current_pos(G); 

edge_array<list_item> worklist_it(G, nil); 

i=0; 

forall(e, edges) 

{ current_order[i]=e; 
current_pos[e]=i++; 

} 

for (i=0; i < edges.size()-1; i++) 

{ e=current_order[i]; 
f=current_order[i + 1] ; 

if ( cross(G, x_coord, current_pos, e, f) ) 
worklist_it[e] = work_list.append(e); 

} 


while (!work_list.empty() ) 

{ e=work_list.pop(); 
worklist_it[e]= nil; 
i=current_pos[e]; 
result++; 

if (i < edges.size 0-1) 

{ f=current_order[i+1]; 
current_order[i]=f; 
current_order[i+1]=e; 
current_pos[e]++; 
current_pos[f]—; 

if (lit=worklist it[f]) 
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{ work_list. del_item (lit) ; 
worklist_it[f]= nil; 

} 

// Test, whether e or f cross their new neighbor: 
if (i>0) 

{ g=current_order[i-1]; 

if (cross(G, x_coord, current_pos, g, f) ) 
{ if (!worklist_it[g]) 

worklist_it[g]= work_list.append(g) ; 

} 

else 

if (lit=worklist_it[g]) 

{ work_list.del_item(lit); 
worklist_it[g]= nil; 

} 

} 

if (i < edges.size 0-2) 

{ h=current_order[i+2] ; 

if (cross(G, x_coord, current_pos, e, h) ) 
worklist_it[e]=work_list.append(e); 

} 

} 

} 


return result; 

} 


void init_positions(graph &G, array<list<node> > &Level, 

node_array<int> &x_coord, node_array<int> &nlabel, 
node_array<int> &x_prio, bool first) 

{ 

node V; 

int i=-lf label=l, x, N=G.number of nodes(); 


while ( (++i) <= max_level) 

{ x=0 ; 

forall(v, Level[i]) 

{ if (first) x_coord[v]=x++; 
if (x_prio[v] == -1) 

{ x_prio[v]= N; 
nlabel[v]= 0; 

} 


else 

{ x_prio[v]= G.degree(v); 
nlabel[v]= label+t; 

} 


} 


} 


} 


// dummy node = highest priority 


bool make_hierarchy (graph &G, node_array<int> &the_level, 
array<list<node> > 
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&Level, list<node> &dummy_nodes, bool first) 

{ 

int i=0; 
node v; 

forall_nodes(v, G) 

Level[ the_level[v] ].append(v); 

if (!first) return false; 


bool lost_edge=false; 
int j ; 
node vj, a, b; 
edge e; 

list<edge> remove, split, turn; 

Make_Simple(G); 

forall_nodes(v, G) 

{ i=the_level[v] ; 

forall_out_edges(e, v) 

{ j=the_level[ G.target(e) ]; 
if (i>j) turn.append(e) ; 
if (i==j) remove.append(e) ; 
if (i+l<j) split.append(e); 

} 


forall(e, turn) 

{ 

a=G.source(e); b=G.target(e) ; 
e=G.rev_edge(e); 

if (the_level[b]+1 < the_level[a] ) split.append(e) ; 

} 

if (!remove.empty()) 

{ error_handler(1,"input is not a hierarchical graph."); 
lost_edge=true; 

forall(e, remove) G.del_edge(e) ; 

} 


duinmy_nodes . clear () ; 

forall(e,split) 

{ 

a = G.source (e); 
b = G.target(e); 

i = the_level[a] + 1; 
j = the_level[b]; 

w = G.new_node(); 
Level[i].append(w); 
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duminy_nodes . append (w) ; 

G.move_edge(e,a, w) ; 
a = w; 
i++; 

while (i<j) 

{ w=G.new_node(); 

Level[i].append(w); 
duininy_nodes . append (w) ; 
G.new_edge(a^ w); 
a = w; 
i++; 

} 

G.new_edge(a,b); 


if (split.empty() ) return lost_edge; 

the_level.init(G); // there are more nodes now 

i=-l; 

while ( (++i) <= max_level) 

forall(v, Level [i]) the_level[v]=i; 

return lost_edge; 

} 


inline int all_crossings(const graph &G, array<list<node> > &Level, 

node_array<int> &x_coord, array<int> &L_crosses) 

{ 

int i=0, crosses=0, c; 
while(i < max_level) 

{ c= L_crosses[i]= number_of_crossings(G, Level, i++, x_coord); 
crosses+=c; 

} 

return crosses; 

} 


inline void copy_all_xcoord(const graph &G, array<list<node> >& /* Level 


&x_new, 

&L_crosses_new) 

{ 


node_array<int> &x_coord, node_array<int> 
array<int> &L_crosses, array<int> 


int i=0; 
node v; 

while(i < max_level) 

{ L_crosses_new[i]= L_crosses[i]; 
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i++; 

} 

forall_nocies (v, G) x_new[v] =x_coord [v] ; 


bool update_best(const graph &G, array<list<node> > &Level, 

node_array<int> &x_coord, node_array<int> &x_new, 

array<int> &L_crosses, const int &i, 

int &crosses, const bitarray &what_to do) 

{ bool update, process_next; 

int rcnl, rcn2=0, next_level, cr_suin, work_level; 

// int make_worse= 1.2; 
node v; 

two_level (G, Level, i, x_coord, x_coord, what_to_do); 

// makes the requested change 


if (what_to_do & SORT_LOWER) 

{ next_level=i+l; 
work_level=i+l / 

} 

else 

{ next_level=i-l; 
work_level=i; 

} 

process_next=((what_to_do & SORT_LOWER) && (next_level < max_level)); 
if ( (what_to_do & SORT_UPPER) && (next_level > -1)) 
process_next=true; 

// what level is neighbor to the changed 

if any 

rcnl= nuiTiber_of_crossings (G, Level, i, x_coord) ; 
if (process_next) 

rcn2= number_of_crossings(G, Level, next_level, x_coord) / 
cr_sum= L_crosses[i]; 

if (process_next) cr_sum+= Lacrosses [next_level] ; 

update= ((rcnl+rcn2) <= cr_sum); 

// update= ((rcnl+rcn2) <= cr_sum* make_worse); 


if (update) 

{ crosses^ crosses- L_crosses[i]+ rcnl; 

L_crosses[i]= rcnl; 

forall(v. Level[work_level]) x_new[v]=x_coord[v]; 
if (process_next) 

{ crosses= crosses- L_crosses[next_level] + rcn2; 
L_crosses[next_level]= rcn2; 

} 

} 

else 

forall(v. Level[work_level]) x_coord[v]=x_new[v]; 
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return update; 

} 


bool down_up_sort(graph &G, array<list<node> > &Level, node_array<int>& 
x_coord, 




array<int> &L crosses, int &crosses, 

node array<int>& /* nlabel */, 

int how often, int first, int begin) 

int 

i=0, 

j=0; 

int 

best 

crosses, orig crosses; 

char 

headline[200]; 

bool 

ch, 

u changes=false, 1 changes=false; 

bool 

result=false; 


node_array<int> x_new(G) ; 
node_array<int> x_old(G); 
array<int> L_crosses_new(max_level) ; 
array<int> L_crosses_old(max_level); 

copy_all_xcoord(G, Level, x_coord, x_old, L_crosses, L_crosses_old); 

array<bool> Lu_changes(max_level) / 
array<bool> Ll_changes (max__level) / 
for(1=0; i < max_level; i++) 

Lu_changes[i]=Ll_changes[i]=false; 

best__crosses= orig_crosses= crosses; 


j=how_often; 
while (j--) 

{ if {best_crosses==0) break; 
if (first==DOWN) 

{ for (i=begin; i< max_level; i++) 

{ ch=two_level (G, Level, i, x_coord, x__coord, SORT_LOWER) ; 
if (ch && !Ll_changes[i]) l_changes=true; 

Ll_changes[i]=ch; 
if (ch) . 


{ crosses-"= L_crosses [i] ; 

L_crosses [i] = nuiTiber_of_crossings (G, Level, i, x_coord) ; 
crosses+= Lacrosses[i] ; 
if (i< max_level-l) 

{ crosses-= L_crosses[i+1]; 

L_crosses[i+1]= number_of_crossings(G, Level, i+1, x_coord); 
crosses+= L_crosses[i+1 ] ; 

} 


if (crosses < best_crosses) 

{ copy_all_xcoord(G, Level, x_coord, x_new. 

Lacrosses, L_crosses_new); 

best_crosses=crosses; // new high score 

} 


} 
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// set 'begin* for up- 


} 

begin=inax_level“l ; 
sort new 
} 

if (best crosses==0) break; 


for (i=begin; i>=0; i—) 

{ ch=two_level(G, Level, i, x_coord, x_coord, SORT_UPPER); 
if (ch && !Lu_changes[i]) u_changes=true; 

Lu_changes[i]=ch; 
if (ch) 

{ crosses-= L_crosses[i] ; 

L_crosses [i] = nuinber_of_crossings (G, Level, i, x_coord) ; 
crosses+= L_crosses[i]; 
if (i>0) 

{ crosses-= Lacrosses[i-1 ]/ 

L_crosses[i-1]= number_of_crossings(G, Level, i-1, x_coord); 
crosses+= L_crosses[i-1]; 

} 

if (crosses < best_crosses) 

{ copy_all_xcoord(G, Level, x_coord, x_new, L_crosses, 
L_crosses_new) ; 

best_crosses=crosses; 

} 

} 


first=DOWN; 
begin=0; 

if ( !(u_changes || l^changes) ) break; else result=true; 

u_changes= l_changes= false; 


if (best_crosses < orig_crosses) 

{ copy_all_xcoord (G, Level, x_new, x^coord, L_crosses_new, L_crosses); 
crosses=best_crosses; 

} 

if (crosses > orig_crosses) 

{ copy_all_xcoord(G, Level, x_old, x_coord, L_crosses_old, L_crosses); 
crosses=orig_crosses; 

} 

return result; 

} 


int 

{ 


down_up_change(graph &G, array<list<node> > &Level, node_array<int> 

&x_coord, node___array<int> &nlabel, bool first) 
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bool skip_sort=true; 

int i=0, j=0; 

int crosses, prev_crosses, pprev_crosses, orig_crosses; 

char headline[200] ; 

node_array<int> x_new(G); 
node_array<int> x_old(G); 
array<int> L_crosses(max_level); 

array<int> L_crosses_new(max_level); 

array<int> L_crosses__old (max_level) ; 

prev_crosses= orig_crosses- crosses= 

all_crossings(G, Level, x_coord, L_crosses); 
pprev_crosses= prev_crosses+l; 

if (first) 

{ down_up_sort (G, Level, x_coord, L_crosses, 

crosses, nlabel, sort_max, DOWN, 0)/ 

#ifdef _DEBUG_SUGI 

sprintf(headline, "hierarchy after normal sort: %d crossings", 
crosses)/ 

if (!draw_hierarchy(G, Level, x_coord, nlabel, headline, true)) 
user_exit(); 

#endif 

} 

if (crosses==0) return 0/ 

copy_all_xcoord(G, Level, x_coord, x_old, L_crosses, L_crosses_old); 
copy_all_xcoord(G, Level, x_coord, x_new, L_crosses, L_crosses_new); 


j=perm_max; 
while (j--) 

{ if (!skip_sort) 

{ down_up_sort(G, Level, x_coord, L_crosses, 

crosses, nlabel, 1, DOWN, 0); 
copy_all_xcoord(G, Level, x_coord, x_new, 

L_crosses, L_crosses_new); // update x_new 

} 

i=0; skip_sort=false; 

while (i < max_level) // down loop 

{ update_best (G, Level, x__coord, x_new, 

L_crosses, i, crosses, S0RT_L0WER | PERMUTE); 
update_best(G, Level, x_coord, x_new, 

L_crosses, i, crosses, S0RT_L0WER); 
if (crosses==0) return 0/ 
i++; 

} 

#ifdef _DEBUG_SUGI 

sprintf(headline, "hierarchy after down-permute, pass %d: %d 
crossings", 

perm__max-j, crosses); 

if (!draw_hierarchy(G, Level, x_coord, nlabel, headline, true)) 
user_exit()/ 

#endif 
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// up loop 


i= max_level-l/ 
while (i>=0) 

{ update_best(G, Level, x_coord, x_new, 

L_crosses, i, crosses, SORT_UPPER | PERMUTE); 
update_best(G, Level, x_coord, x_new, 

L_crosses, i, crosses, SORT_UPPER); 
if (crosses==0) return 0; 
i—/ 

} 

#ifdef _DEBUG_SUGI 

sprintf(headline, "hierarchy after up-permute, pass %d: %d 
crossings", 

perm_max-j, crosses); 

if (!draw_hierarchy(G, Level, x_coord, nlabel, headline, true)) 
user_exit () ; 

#endif 

if ((pprev_crosses== crosses) && (prev_crosses== crosses)) 
break; 

pprev_crosses= prev_crosses; prev__crosses= crosses; 

} // outer loop 


// do_debug=t rue; 

down_up_sort(G, Level, x_coord, L_crosses, 

crosses, nlabel, 2, DOWN, 0); 
if (crosses > orig_crosses) 

{ copy__all_xcoord (G, Level, x_old, x_coord, L_crosses_old, L_crosses) ; 
return orig_crosses; 

} 

return crosses; 

} 


int improve_coord(graph &G, node_array<int> &the_level, array<list<node> 
> ~ 


&x_prio, 

{ 


&Level, node_array<int> &x__coord, 
node_array<int>& /*nlabel */) 


int i, x_min=100, x_max=-100, width=0, wl=0; 

int XV, xw, epos, Ipos, rpos, y_prev; 

node V, w; 
edge e; 

list_item lit; 

node_array<list_item> LIT(G); 


node_array<int> 


G.sort nodes(x coord); 
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i=0; 

while (i <= max_level) Level [i4--}-] . clear () ; 

forall_nodes(v, G) LIT[v]=Level[ the_level[v] ].append(v); 
i=0; 

while (i < max_level) 

two_level(G, Level, i++, x_coord, x_prio, SORT_LOWER | IMPROVE); 
#ifdef _DEBUG_SUGI 

if (!draw_hierarchy(G, Level, x_coord, nlabel, "1st down improving", 
true)) 

user_exit(); 

#endif 


while (i > 0) 

two_level(G, Level, —i, x_coord, x_prio, SORT_UPPER | IMPROVE); 
#ifdef _DEBUG_SUGI 

if (!draw_hierarchy(G, Level, x_coord, nlabel, "1st up improving", 
true)) 

user_exit(); 

#endif 


i=“l; 

while ((++i) <= max^level) 

{ if (Level[i] .size() > width) 

{ width=Level[i].size(); wl=i; } 

} 

while (wl < max_level) 

two_level(G, Level, wl++, x_coord, x_prio, SORT_LOWER | IMPROVE); 
#ifdef _DEBUG_SUGI 

if (!draw_hierarchy(G, Level, x_coord, nlabel, "2nd down improving", 
true) ) 

user_exit(); 

#endif 


forall_nodes(v, G) 

{ cpos= lpos= rpos= 0; 

forall_out_edges(e, v) 

{ w=G.target(e); 

xv=x_coord[v] ; xw=x_coord[w] ; 
if (xv == xw) cpos++; 

if (xv+1 == xw) rpos++; // an edge right slanting 

if (xv-1 == xw) lpos++; 


forall_in_edges(e, v) 

{ w=G.source(e); 

xv=x_coord[v]; xw=x_coord[w]; 
if (xv == xw) cpos+t; 
if (xv+1 == xw) rpos++; 
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if (xv-1 == xw) lpos++; 


if ((cpos>=rpos) && (cpos>=lpos)) continue; 
lit=LIT[v]; i=the_level[v]; 
if (lpos> rpos) 

{ lit=Level[i] .pred(lit) ; 

if (lit) xw=x_coord[ Level[i].inf(lit) ]; 
if (!lit II (xw 1= xv-1)) x_coord[v]—; 

} 

else 

{ lit=Level[i].succ(lit); 

if (lit) xw=x_coord[ Level[i].inf(lit) ]; 
if (!lit II (xw != xv+1)) x_coord[v]++; 

} 

} 


G.sort^nodes(x_coord); 

i=0; y_prev=x_coord[G.first_node()] ; 

forall_nodes(v, G) 
if (x_coord[v] == y_prev) 

{ y_prev= x_coord[v]; x_coord[v]=i; } 

else 

{ y_prev= x_coord[v]; x_coord[v]=++i; } 

return i+1; 

} 


int SUGIYAMA_and_info(graph &G, node_array<int> &x_coord, 

node_array<int> &the_level, list<node> 


& duinmy_node s, 


bool first= true) 


{ 


if (G. nuinber_of_nodes () == 0) // paranoia setting 

return 0; 

int crosses; 
node v; 

max_level= 0; 
forall_nodes(v, G) 

if (the_level [v] > inax_level) inax_level= the_level [v] ; 

array<list<node> > Level(max_level+l) ; 

if (inake_hierarchy(G, the_level. Level, duinmy_nodes, first) ) 
return -1; 

if (first) X coord.init(G); 
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node_array<int> nlabel(G); 
node_array<int> x_prio(G, 0) ; 

forall(v, dummy_nodes) x_prio[v]= -1; 

init_positions(G, Level, x_coord, nlabel, x_prio, first); 

#ifdef _DEBUG_SUGI 
int i; 
char head[200]; 

i=crosses=0; 

while(i < max_level) 

crosses+= number_of_crossings(G, Level, i+t, x_coord); 
sprintf(head, "original hierarchy: %d crossings", crosses); 
if (!draw_hierarchy(G, Level, x_coord, nlabel, head, true)) 
user_exit(); 

#endif 


crosses= down_up_change(G, Level, x_coord, nlabel, first); 

#ifdef _DEBUG_SUGI 

sprintf(head, "Sort/permute finished with %d crossings", crosses); 

if (!draw_hierarchy(G, Level, x_coord, nlabel, head, true)) 
user_exit(); 

#endif 

int width= improve_coord(G, the_level. Level, x_coord, x_prio, nlabel) 
#ifdef _DEBUG_SUGI 

if (!draw_hierarchy(G, Level, x_coord, nlabel, "final embedding", 
true)) 

user__exit () ; 

#endif 


return crosses; 

} 


int SUGIYAMA_and_info(graph &G, node_array<int> &x_coord, 

node_array<int> &the_level, list<node> 


&dummy_nodes, 


bool first, char ^headline, file ostream 


&ps_out) 


if (G.number_of_nodes() == 0) // paranoia setting 

return 0; 

int crosses; 
node v; 

max_level= 0; 
forall_nodes(v, G) 

if (the level[v] > max level) max_level= the_level[v]; 
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array<list<node> > Level(max_level+l); 

if (mak;e_hierarchy (G, the_level, Level, dummy_nodes, first) ) 
return -1; 

if (first) X coord.init(G); 


node_array<int> nlabel(G); 
node_array<int> x_prio(G, 0); 

forall(v, duinmy_nodes) x_prio[v]= -1; 

init_positions(G, Level, x_coord, nlabel, x_prio, first); 

#ifdef _DEBUG_SUGI 
int i ; 
char head[200]; 

i=crosses=0; 

while(i < max_level) 

crosses+= nuiTiber_of_crossings (G, Level, i++, x_coord) ; 
sprintf(head, "original hierarchy; %d crossings", crosses); 
if (!draw_hierarchy(G, Level, x_coord, nlabel, head, true)) 
user_exit(); 

#endif 


crosses= down_up_change(G, Level, x_coord, nlabel, first); 

#ifdef _DEBUG_SUGI 

sprintf(head, "Sort/permute finished with %d crossings", crosses); 

if (!draw_hierarchy(G, Level, x_coord, nlabel, head, true)) 
user_exit(); 

#endif 

int width= improve_coord(G, the_level. Level, x_coord, x_prio, nlabel) 
#ifdef _DEBUG_SUGI 

if (!draw_hierarchy(G, Level, x_coord, nlabel, "final embedding", 
true)) 

user_exit(); 

#endif 


write_to_ps(G, Level, x_coord, 

nlabel, width, max_level, 
headline, ps_out); 


return crosses; 

} 


int SUGIYAiyiA_EMBED (graph &G, node_array<int> &x_coord, 

node_array<int> &the_level, list<node> 

&duinmy_nodes) 

{ 

return SUGIYAMA_and_info(G, x_coord, the_level, dummy_nodes, true); 
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} 


int SUGIYAMA_iterate(graph &G, node_array<int> &x_coord, 

node_array<int> &the_level, list<node> 

&dummy_nodes) 

{ 

return SUGIYAMA_and_inf o (G, x_coord, the_level, duinmy_nodes, false); 

} 


int SUGIYAMA_simple(const graph &G, const node_array<int> &the_level) 

{ 

if (G.number_of_nodes() == 0) // paranoia setting 

return 0; 


char fi1ename[200]; 

node v; 
edge e; 

list<node> duinmy__nodes ; 

graph CG; 

node_array<node> Cnode(G); 


forall__nodes (v, G) Cnode [v] =CG. new_node () ; 

forall_edges(e, G) CG.new_edge(Cnode[G.source(e)] , Cnode[G.target(e)] 


node_array<int> x_coord(CG); 
node_array<int> CLevel(CG); 
forall_nodes(v, G) CLevel[Cnode[v]]=the_level[v] ; 

sprintf(filename, "sugi_out.ps"); 
file_ostream ps_out(filename); 


return 


} 


SUGIYAMA_and_info(CG, x_coord, CLevel, 

filename, ps_out); 


dummy_node s, 


true. 


int SUGIYAMA_EMBEDDING (graph &G, node_array<int>& xcoord, 

node_array<int>& level, 
edge_array<list<int> >& xpoly) 

{ 

list<node> dummy_nodes; 

int crossings = SUGIYAMA_and_info (G, xcoord, level, duinmy_nodes, true) 
xpoly.init(G); 

node_array<bool> dummy(G,false) ; 
node v; 

forall (V, duinmy_nodes) dummy [v] = true; 
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forall_nodes(v,G) 

{ if (dummy[v]) continue; 

edge e = G.first_adj_edge(v) ; 
while (e) 

{ edge next = G.adj_succ(e) ; 
edge x = e; 
node u = target(x); 
while (dummy[u]) 

{ xpoly[e].append(xcoord[u] ) ; 

X = G.first_adj_edge(u) ; 
u = target(x); 

} 

if (u != target(e)) 

G.move_edge(e,s ource(e),u); 
e = next; 

) 


forall(V,dummy_nodes) G.del_node(v); 
return crossings; 

} 
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^'k-k'k^'k'k'k'k'k^'tc'k'k^'k-k'k'k-k^'k'fr'k'k'k'k'k'k'tc'k'k'k'k'k-^k'k'k'k'k'k'tc'k'k'k'k'k'k'k'k'k'k^-k'k'k'k-k'k'k'k'kitc^^^k'k'k'k^'k'tc 

+ 

+ LEDA 3.5.1 
+ 

+ _tutte.c 
+ 

+ This file is part of the LEDA research version (LEDA-R) that can be 
+ used free of charge in academic research and teaching. Any commercial 
+ use of this software requires a license which is distributed by the 
+ LEDA Software GmbH, Postfach 151101, 66041 Saarbruecken, ERG 
+ (fax +49 681 31104) . 

+ 

+ Copyright (c) 1991-1997 by Max-Planck-Institut fuer Informatik 
+ Im Stadtwald, 66123 Saarbruecken, Germany 
+ All rights reserved. 

+ 


★ ★-A-Tlr + Tlr'*' ^ 

U - II 

// drawing of a graph using Tutte’s algorithm // 

// David Alberts (1996) // 

// - // 


ttinclude <LEDA/graph_alg.h> 
#include <LEDA/vector.h> 
ttinclude <LEDA/matrix.h> 


bool TUTTE_EMBEDDING(const graphs G, const list<node>& fixed_nodes, 

node_array<double>& xpos, node_array<double>& ypos) 

{ 

// computes a convex drawing of the graph G if possible. The list 
// fixed_nodes contains nodes with prescribed coordinates already 
// given in pos. The computed node positions of the other nodes are 
// stored in pos, too. If the operation is successful, true is 
returned. 

// Precondition: pos is valid for G. 


node v,w; 
edge e; 

node_array<bool> is_fixed(G,false)/ 
forall(v,fixed nodes) is fixed[v] = true; 


list<node> other_nodes/ 

forall_nodes(v,G) if(!is_fixed[v]) other_nodes.append (v) ; 
node_array<int> ind(G); // position of v in other_nodes and A 

int i=0; 

forall(v,other nodes) ind[v] = i++; 


int n = other_nodes.size(); 
vector coord(n); 
vector rhs(n); 
matrix A(n,n); 


// #other nodes 

// coordinates (first x then y) 
// right hand side 
// equations 


// initialize non-zero entries in matrix A 
forall(v,other_nodes) 

{ 

double one_over_d = 1.0/double(G.degree(v)); 
forall_inout_edges(e,v) 

{ 
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// get second node of e 

w = (v == source(e)) ? target(e) : source(e); 

if(!is_fixed [w]) A (ind[v],ind[w]) = one_over_d; 

} 

A(ind[v],ind[v]) = -1; 


if{!A.det()) return false; 

// compute right hand side for x coordinates 
forall(V,other_nodes) 

{ 

rhs[ind[v]] = 0; 

double one__over__d = 1.0/double (G. degree (v) ) ; 
forall_inout_edges(e,v) 

{ 

// get second node of e 

w = (v == source(e)) ? target(e) : source(e); 

if(is_fixed[w] ) rhs[ind[v]] -= (one_over_d*xpos[w]) 

} 


// compute X coordinates 
coord = A.solve(rhs); 

forall(v,other_nodes) xpos[v] = coord[ind[v] ] ; 

// compute right hand side for y coordinates 
forall(V,other_nodes) 

{ 

rhs[ind[v]] = 0; 

double one_over_d = 1.0/double{G.degree (v) ) ; 
forall_inout_edges(e,v) 

{ 

// get second node of e 

w = {v == source(e)) ? target(e) : source(e); 

if(is_fixed[w]) rhs[ind[v]] -= (one_over_d*ypos[w]) 

} 


// compute y coordinates 
coord = A.solve(rhs); 

forall(v,other_nodes) ypos[v] = coord[ind[v] ] ; 
return true; 
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APPENDIX B: CODE 


TEST CODE CREATED 


#include ”graph_layout.h" 

// 

// MAIN PROGRAM 

// 

main () 

{ 

GRAPH<int,int> G; 

init_graph(G, true); 

run_ortho(G, true); 
run_straight_line(G, true); 
run_straight_line2(G, true); 
run_tutte(G, true); 
run_spring_embedding(G, true); 
run_d2_spring_erribedding (G, true); 

return 0; 
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B, 


NEW CODE CREATED 


#include <LEDA/graph.h> 
#include <LEDA/graph_alg.h> 


#ifndef GRAPH_LAYOUT_H 
#define GRAPH_LAYOUT_H 1 

void run_ortho(graphs, bool); 

void run_straight_line(graphs, bool); 

void run_straight_line2(graphs, bool); 

void run_tutte(graphs, bool); 

void run_d2_spring_embedding(graphs, bool) 

void run_spring_embedding(graphs, bool); 

void init_graph(graphs, bool); 

#endif 
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#include <fstream.h> 

#include <iostream.h> 

#include <stdlib.h> 

#include <stream.h> 

#include "graph_layout.h" 

// 

// THIS IS THE ORTHOGONAL LAYOUT ALGO 

// 

void run_ortho(graphs G, bool DEBUG) 

{ 

node v; 

node_array<int> x(G),y{G); 

edge_array< list<int> > xbends(G), ybends(G); 

if (DEBUG) 

{ 

cout « "NO EMBEDDING 
newline; 

} 

forall_nodes(v,G) 

{ 

x[v] = rand_int(1,500); 
y[v] = rand_int(1,500); 

} 

if (DEBUG) 

{ 

forall_nodes(V,G) cout « string("x = %3d y = %3d\n",x[v],y[v]); 

cout « "ORTHO EMBEDDING 

newline; 

} 

ORTHO_EMBEDDING(G, x, y, xbends, ybends, false); 

if (DEBUG) 

{ 

forall_nodes(V,G) cout « string(”x = %3d y = %3d\n",x[v],y[v]) ; 
G.write_gml("graph.ortho"); 

} 

} 

// 

// THIS IS THE STRAIGHT_LINE_2 ALGO 

// 

void run_straight_line2(graphs G, bool DEBUG) 

{ 

node v; 

node_array<int> x(G)/y(G); 

if (DEBUG) 

{ 

cout « "NO EMBEDDING 
newline; 

} 

forall_nodes(v,G) 

{ 

x[v] = rand_int(1,500); 
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y[v] = ranci_int (1,500) ; 

) 

if (DEBUG) 

{ 

forall_nodes (V, G) cout << stringC'x = %3d y = %3d\n", x [ v] , y [ v] ) ; 

cout « "STRAIGHT LINE EMBEDDING2 

newline; 

) 

STRAIGHT_LINE_EMBEDDING2(G, x, y) ; 

if (DEBUG) 

{ 

forall_nodes(V,G) cout << string("x = %3d y = %3d\n",x[v],y[v]); 
G.write_gml("graph.straight_12"); 

) 

) 

// 

// THIS IS THE STRAIGHT LINE ALGO 

// 

void run_straight_line(graphs G, bool DEBUG) 

{ 

node v; 

node_array<int> x(G),y(G); 

if (DEBUG) 

{ 

cout « "NO EMBEDDING 
newline; 

} 

forall_nodes(v,G) 

{ 

x[v] = rand_int(1,500) ; 
y[v] = rand_int(1,500) ; 

} 

if (DEBUG) 

{ 

forall_nodes (V, G) cout << stringC'x = %3d y = %3d\n", x [v] , y [v] ) ; 

cout « "STRAIGHT LINE EMBEDDING 

newline; 

} 

STRAIGHT_LINE_EMBEDDING(G, x, y) ; 

if (DEBUG) 

{ 

forall_nodes (V, G) cout « stringC'x = %3d y = %3d\n", x [v] , y [v] ) ; 
G.write_gml("graph.straight_l" ) ; 

} 

) 

// 

// THIS IS THE TUTTE ALGO 

// 

void run_tutte(graphs G, bool DEBUG) 

{ 

node_array<double> dx(G), dy(G); 
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list<node> L; 
node v; 

if (DEBUG) 

{ 

cout « "NO EMBEDDING 
newline; 

} 


forall_nodes(v,G) 

{ 

dx[v] = (double) rand_int(1, 500); 
dy[v] = (double) rand_int(1, 500) ; 

) 

if (DEBUG) 

{ 

forall_nodes(v,G) cout << string("x = %lf y = %lf\n",dx[v],dy[v]) 

cout « "TUTTE EMBEDDING 

newline; 

} 


TUTTE_EMBEDDING(G,L,dx,dy) ; 

if (DEBUG) 

{ 

forall_nodes (v, G) cout « stringC'x = %lf y = %lf\n", dx[v] , dy [v] ) 
G. write_ginl ("graph. tutte") ; 

} 

} 

// 

// THIS IS THE D2 SPRING EMBEDDING ALGO 

// 

void run_d2_spring_erabedding(graphs G, bool DEBUG) 

{ 

node_array<double> dx(G), dy(G); 
node v; 

if (DEBUG) 

{ 

cout « "NO EMBEDDING "; 

newline; 

) 

forall_nodes(v,G) 

{ 

dx[v] = (double) rand_int(1,500); 
dy[v] = (double) rand_int(1, 500); 

> 

if (DEBUG) 

{ 

forall_nodes (V, G) cout << stringC'x = %lf y = %lf \n", dx [v] , dy [v] ) 

cout « "D2 SPRING EMBEDDING EMBEDDING "; 

newline; 

} 


D2_SPRING_EMBEDDING(G,dx,dy,0.0, 500.0,0.0,500.0, 400) ; 
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y = %lf\n",dx[v],dy[v]) 


if (DEBUG) 

{ 

forall_nodes(V,G) cout << string("x = %lf 
G.write_gml("graph.d2_spr_emb"); 

} 

} 

// 

// THIS IS THE SPRING EMBEDDING ALGO 

// 

void run_spring_eiribedding (graphs G, bool DEBUG) 

{ 

node_array<double> dx(G), dy(G); 
node v; 

if (DEBUG) 

{ 

cout « "NO EMBEDDING 
newline; 

} 

forall_nodes(v, G) 

{ 

dx[v] = (double) rand_int(1, 500) ; 
ciy[v] = (double) rand_int (1,500)/ 

} 

if (DEBUG) 

{ 

forall_nodes(V,G) cout << string("x = %lf y = %lf\n",dx[v],dy[v]) 

cout « "SPRING EMBEDDING EMBEDDING 

newline; 

} 


SPRING_EMBEDDING(G,dx,dy,0.0,500.0, 0.0, 500.0, 400); 

if (DEBUG) 

{ 

forall_nodes (V, G) cout << string ("x = %lf y = %lf \n", dx [v] , dy [v] ) 
G. write_gml ( "graph. spr_emb" ) ; 

} 

} 

// 

// INITIALIZE GRAPH TO BIDIRECTIONAL AND PLANAR 

// 

void init_graph(graphs G, bool DEBUG) 

{ 

if (DEBUG) 

{ 

cout << "Inside init_graph." ; 
newline; 

} 

int n = read_int("n = "); 
random_planar_graph(G,n,n*2); 
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list<edge> bi_edges = Make Bidirected (G); 
G. inake_map () ; “ ' 

G.make_planar_map() ; 

} 
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c. 


CAPS CODE MODIFIED 


/* 


NOTES: 


*/ 


I -k •k-k-k-k-k'k-k'k'k'k^'k'k'k'k'k'k'k'k-k-k-k'k-k'k^-k'k'k-k-k'k-k-k-k-k-k'k-k-k-k-kk-k-k-k-kk-k'k'k-k-k-k-k-k-k'k-k-k-k 

Name: graph_editor.C 

Author: Capt Robert M. Dixon 

Program: graph_editor 

Date Modified: 21 Sep 92 

Remarks: graph_editor.C is the main program for the 

CAPS *93 graph editor. Depending on the command line 

parameters, it allows either viewing only or full 
editing of a graph passed by the CAPS * 93 syntax- 
directed editor. 

General Comments: 

The XmProcessTraversal function is called numerous 
places in an attempt to keep the keyboard input focus 
in the drawing window. This allows the editor to 
respond to the delete and backspace key. This works 
with varying degrees of success. 

Credits: Portions of code are adapted from the following: 

Barakati, Naba, X Window System Programming, SAMS, 
1991. 


Heller, Dan, Motif Programming Manual, O'Reilly and 
Associates, 1991. 


Johnson, Eric, and Reichard, Kevin, X Window 
Applications Programming, MIS Press, 1989. 

Young, Douglas, Object Oriented Programming With C++ 
and OSF/Motif, Prentice-Hall, 1992. 


Reengineering: 

Modified by Doug Lange on 8/12/96 

Removed Property button and put in its place a Timers 


button 


button. 


Modified by Doug Lange 
Added callback 
Modified by Doug Lange 
Added callback 


on 8/16/96 - 8/20/96 

and dialog for Timer Tool button. 

on 8/19/96 

and dialog for Informal Description Tool 


History: 

@1 96/09/29 Ken Moeller 
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Migration from Motif 1.2 to Motif 1.1. 


@2 

@3 

04 

@5 


96/10/01 Ken Moeller 

Upgraded calling arguments to reflect changes to requirements, 
96/10/03 Ken Moeller 

Started to switch over to build_from_sde and write_to_sde. 

This is not yet complete. 

96/10/03 Ken Moeller 

Items removed or tests added while testing @3. 


96/10/04 Ken Moeller 
Removal of viewer code. This option is no longer supported. 
Still need to investigate the resources. So the job is not 
complete. 

@6 96/10/06 Ken Moeller 

Change in how units are encoded. 

@7 96/10/11 Ken Moeller 

Moved print command over to an XEvent so that the window 
can be refreshed before the screen is captured. 




#include 

#include 

#include 

#include 

#include 

#include 


<fstream.h> 
<iostream.h> 
<stdlib.h> 
<stream.h> 
<sys/stat.h> 
<sys/types.h> 


#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

tinclude 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 


<Xll/Xatom.h> 

<X11/cursorfont.h> 
<Xll/keysym.h> 
<Xm/DialogS.h> 
<Xm/DrawingA.h> 
<Xm/DrawnB.h> 
<Xm/Form.h> 
<Xm/LabelG.h> 
<Xm/List.h> 
<Xm/MainW.h> 
<Xm/MessageB.h> 
<Xm/PanedW.h> 
<Xm/PushB.h> 
<Xm/PushBG.h> 
<Xm/RowColumn.h> 
<Xm/ScrolledW.h> 
<Xm/SelectioB.h> 
<Xm/Separator.h> 
<Xm/Text.h> 
<Xm/TextF.h> 
<Xm/ToggleBG.h> 


//#include "ge_utilities_debug.h" 
#include <stdio.h> 

#include <math.h> 


#include "action_area.h" 
#include ”build_option.h” 


//Added by DL 8/19/96 
//Added by DL 8/19/96 


//Added by DMA 9/18/96 
//Added by DHA 9/18/96 


//Added by DL 8/19/96 


//Added by DL 8/19/96 
//Added by DL 8/16/96 
//Added by DL 8/19/96 


//Added by DHA 8/20/96 


//Added by DL 8/19/96 
//Added by DHA 8/20/96 
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#include ”get_unique_id.h” 
#include ”ge_defs.h” 

#include *’ge_interface . h” 
#include "gettopshell.h” 
#include "graph_editor.h” 
#include ”graph_object_list.h" 


//Added by MY 8/4/97 
//Added by DL 8/16/96 
//Added by DL 8/16/96 
//Added by DHA 8/19/96 
// kbm 


//#include 

9/12/96 

"op prop exception.h" 

//Added 

for 

req. 

7.4, dha 

//#include 

9/15/96 

"op prop formal desc.h" 

//Added 

for 

req. 

7.6, dha 

//ttinclude 

"op prop informal desc.h” 

//Added 

for 

req. 

7 dha 9/15/96 

//#include 

9/15/96 

"op prop keywords.h" 

//Added 

for 

req. 

7.6, dha 

//#include 

9/12/96 

"op prop output guard.h" 

//Added 

for 

req. 

7.3, dha 

//#include 

9/12/96 

"op prop timer op.h” 

//Added 

for 

req. 

7.5, dha 

//#include 

9/5/96 

"op prop_timing_info.h” 

//Added 

for 

req. 

7 & 7.7, dha 

//ttinclude 

9/12/96 

"op prop trigger cond.h” 

//Added 

for 

req. 

7.2, dha 


#include 
#include 
ttinclude 
#include 
#include 
ttinclude 
#include 
ttinclude 
#include 
#include 
#include 
#include 
#include 


”operator_obj ect.h” 
"postpopup.h” 

"setcursor . h*' 
”spline_object.h” 
”stream_obj ect.h” 
”operator_property_menu.h" 
”stream_property_menu.h" 
”timer_tool.h" 

"windows.h" 

"warning.h” 

"ge_utilities_debug.h" 
"report_errors.h” 
”graph_layout.h” 


//Added by DHA 8/20/96 
//Added by DHA 8/15/96 


//Added for req. 8, dha 9/16/96 
//Added by DL 8/22/96 
// kbm 


// Added by BR 9/6/97 


// MAXCOLORS is the number of colors defined to the editor. 
// To add or subtract colors, this value must be modified. 


// changed size from 75 
#define BUTTONWIDTH 65 
#define HELPSIZ 1000 


// graph_editor has a number of global variables due to 
// Motif’s use of callback functions. Since these functions 
// have fixed formal parameter lists, global variables must 
// be used to pass some data between functions. 

// All drawing commands are executed on both drawing_a and 
// drawing_area_pixmap. drawing_a is the visible canvas, while 
// drawing_area_pixmap provides a backup. When the canvas needs 
// to be redrawn, the drawing in drawing_area pixmap is merely 
// copied back onto the canvas. 

// colors[] is a list of predefined X colors. To use others, 

// consult an X reference giving allowable color names. Using 
// the predefined colors allows the user to specify color 
// preferences in X resource text files. 

// graphic_list is a GraphObjectList containing all the 
// visible operators and streams. 


242 


// selected^object_ptr always points to the object selected 
// (i. e. with handles around it) on the drawing canvas. 

// nuin_del_ops is the number of deleted operators, and 
// del__op_id is an array of identifiers for deleted operators. 

// The Resrcs struct, resources[], and options[] are used 
// by Motif for parsing the command line options. 


Widget toplevel, main_w, menubar, rowcol, scrolled_win, 

op__button, term_button, stream_button, select_button, 
spec_button, informal^button, types_button, 
timers_button, button_divider; 

XtAppContext app; 

Pixmap op_button_pixmap, term__button_pixmap, stream_button_pixmap, 
select_button_pixmap, spec_button_pixmap, 
informal_button_pixmap, 

types__button_pixmap, 

timers_button_pixmap; 

XGCValues gcvl, gcv2, gcv3/ 

Screen *screen_ptr; 

XtActionsRec actions; 

String translations = 

”<BtnlDown>: draw(down)\n\ 

<BtnlUp>: draw(up) \n\ 

<BtnlMotion>: draw(motion)\n\ 

<Btn3Down>: draw{btn3down)\n\ 

<Btn3Motion>: draw{btn3motion) \n\ 

<Btn3Up>: draw(btn3up)\n\ 

<MotionNotify>: draw(motionnotify) \n\ 

<Key>: draw{key)\n\ 

<Key>Tab: draw(tab)"/ 

unsigned long gc_mask/ 

Window root_window, toplevel_window; 

// XEvent *print_event = (XEvent *) malloc(sizeof(XEvent) ) ; // @7 

XEvent *print_event; 

extern int Global_argc; 
extern char **Global_argv; 

// MY 8/5/97 

extern int enter_operator; 
extern int enter_stream; 
extern int enter_errs; 
int enter_types; 
int enter_spec; 
int enter_timer; 
int enter_inform; 

// MY 8/5/97 

void types close_dialog(Widget w, XtPointer client_data, XtPointer 
call__data) { 

enter_types = 0; 

close_dialog(w, client_data, call_data) / 


void spec_close_dialog(Widget w, XtPointer client_data, XtPointer 
call_data) { 

enter_spec = 0; 

close_dialog(w, client_data, call_data) ; 
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} 


void timer_close_dialog(Widget w, XtPointer client_data, XtPointer 
call_data) { 

enter_timer = 0; 

close_dialog(w, client_data, call_data); 


void inform_close_dialog(Widget w, XtPointer client_data, XtPointer 
call_data) { 

enter__inf orm = 0; 

close_dialog(w, client_data, call_data); 


int still_open() 

{ 

return ( enter types | | enter spec | | enter timer | | enter_inform | | 



int is_empty(char * str) 

{ 

if ( *str == * * ) return 1; 

return 0; 

} 

/★:Ar****************** + ****** + **++************** + **:*r*****************:t + 3t 

ream 

Added by Doug Lange 8/16/96.*/ 

GRAPH_DESC gdnode; 

ID_LIST idp; 

ACTION_NODE* next_action_ptr; // kbm 

GC std_graphics_context, dotted_context, erase^context; 

Dimension width, height; 

Pixmap drawing_area_pixmap; 

Widget drawing_a, current_op_name^ current_op_met; 

Widget save_indicator, error_indicator, status_indicator; 

BOOLEAN state_stream = false, alt_selected = false, ctrl_selected = 
false; 

BOOLEAN ibar_mode = false; // added for req #6.1. dha 
BOOLEAN label_edit_mode = false; // added for req #6.1.1. dha 
//MY 8/4/97 

char default_name[INPUT_LINE_SIZE]; // added for req #6.4 dha 
CLASS_DEF object_def = GRAPHOBJECT; // added for req #6.1. dha 
char* colors[] = {"Aquamarine", "Black", "Blue", "BlueViolet", 

"Brown", "CadetBlue", "Coral", 

"CornflowerBlue", "Cyan", "DarkGreen", 

"DarkOliveGreen", "DarkOrchid", 

"DarkSlateBlue", "DarkSlateGrey", 

"DarkTurquoise", "DimGrey", "Firebrick", 

"ForestGreen", "Gold", "Goldenrod", "Grey", 

"Green", "GreenYellow", "IndianRed", "Khaki", 
"LightBlue", "LightGrey", "LightSteelBlue", 

"LimeGreen", "Magenta", "Maroon", 

"MediumAquamarine", "MediumBlue", 

"MediumOrchid", "MediumSeaGreen", 

"MediumSlateBlue", "MediumSpringGreen", 
"MediumTurquoise", "MediumVioletRed", 

"MidnightBlue", "Navy", "Orange", "OrangeRed", 
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"Orchid”, "PaleGreen”, "Pink”, "Plum”, "Red", 
"Salmon", "SeaGreen", "Sienna”, "SkyBlue”, 
"SlateBlue", "SpringGreen", "SteelBlue”, 

”Tan", "Thistle", "Turquoise”, "Violet", 

"VioletRed", "Wheat", "White", "Yellow", 
"YellowGreen"}; 

unsigned long color_table[MAXCOLORS + 1]; 

TOOL_STATE tool_state = SELECT_TOOL; 

GraphObjectList graphic_list; 

GraphObject* selected_object_ptr = NULL; 

OperatorObject *op_being_updated = NULL; // Add for req. 7, dha 
StreamObject *st_being_updated = NULL; // Add for req. 8, dha 
Display *display__ptr; 

Window draw_window; 

int default_color = WHITE; 

int default_font = COURIERBOLD12; 

int num_del_ops = 0; 

OP_ID del_op_id[MAXDELETEDOPS] ; 

ERROR_MSGS errors^present; 

BOOLEAN psdl_modified, syntax_checked; 

int save_performed; // updated save_state when you return 

char *help_menu_f iles [ ] = { "psdl__grammar.hip", 

"operators.hip", 

"streams.hip”, 

"exceptions.hip", 

"timers.hip"} ; 

// ?? Look at this to see if still needed ***************T*r*********** 
struct _resrcs { 
int viewer; 

} Resrcs; 

static XtResource resources[] = { 

{"viewer", "Viewer", XmRBoolean, sizeof (int), 

XtOffsetOf(struct _resrcs,viewer), XmRImmediate, False), 

}; 


static XrmOptionDescRec options[] = { 

{ff_v", "viewer", XrmoptionNoArg, "True"), 

}; 


//★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★it************ 


void select_state(TOOL_STATE new_state) { 
tool state = new state; 


if (new_state == 
XtVaSetValues 
else 

XtVaSetValues 


= OPERATOR_TOOL) 
(op_button, 

(op_button. 


XmNshadowType, 
XmNshadowType, 


XmSHADOW_IN, 

XmSHADOW_OUT, 

XmSHADOW_IN, 
XmSHADOW OUT, 


if (new_state == TERMINATOR_TOOL) 

XtVaSetValues(term_button, XmNshadowType, 

else 

XtVaSetValues(term_button, XmNshadowType, 

if (new_state == STREAM_TOOL) 

XtVaSetValues(stream_button, 
else 

XtVaSetValues(stream button. 


XmNshadowType, XmSHADOW_IN, 
XmNshadowType, XmSHADOW OUT, 


NULL); 
NULL); 

NULL); 
NULL); 

NULL); 
NULL); 
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if (new_state == SELECT_TOOL) 
XtVaSetValues(select_button, 
else 

XtVaSetValues(select_button, 

// Display all other buttons 
/* ?? delete if not needed 
XtVaSetValues(types_button, 
XtVaSetValues(spec_button, 
XtVaSetValues(timers_button, 
XtVaSetValues(informal_button, 
*/ 


XmNshadowType, XmSHADOW_IN, NULL); 
XmNshadowType, XmSHADOW_OUT, NULL); 


XmSHADOW_OUT, NULL); 
XmSHADOW_OUT, NULL); 
XmSHADOW_OUT, NULL); 
XmSHADOW_OUT, NULL); 


XmNshadowType, 
XmNshadowType, 
XmNshadowType, 
XmNshadowType, 


/•k-k^^ic-k-k-k-k-k-ic-k-k'k-k-k'k'k'k'k-k'k'k-ic-k'k'k'k-ic-k-k-k-k-k-k-k-k-k-k-k-k-k-k^^-k-k-k-k'k-k'k-k'k-k'k-k'k-k'k-k'k-k^'k-k'k'k-k'k'k'k 
* * * * 

* error label() — 


void error label() { 


XmString label; 


/* 

* MY 7/22/97 

★ / 


if ((errors_present == NULL) || 
label = XmStringCreateSimple(” 
XtVaSetValues(error_indicator, 
XtVaSetValues(error_indicator, 


(!syntax_checked)) { 

Check ”); // MY: Check (Syntax) 
XmNlabelString, label, NULL); 
XmNshadowType, XmSHADOW_OUT, NULL); 


} 


else { 

label = XmStringCreateSimple("ERROR MSGS"); 

XtVaSetValues(error_indicator, XmNlabelString, label, NULL); 
XtVaSetValues(error_indicator, XmNshadowType, XmSHADOW_OUT, NULL); 

} 


XmStringFree(label); 

/* 

* unmasked S/6/91 
*/ 

} 


/★★★★★★★★★★★★★★-A-******************************************************** 

* save_state() — Updates the save_indicator with the current indicated 

* state. 

★ ★ ★ y 

void save_state(int state) { 

XmString label; 

if (state == NOT_MODIFIED) { 

label = XmStringCreateSimple("Save Not Required"); 
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XtVaSetValues{save_indicator, XmNlabelString, label, NULL); 
XtVaSetValues{save_indicator, XmNshadowType, XmSHADOW__IN, NULL); 
psdl_modifled = false; 

} 

else if (state == SAVE_REQUIRED) { 

label = XmStringCreateSimple("SAVE REQUIRED”); 

XtVaSetValues(save_indicator, XmNlabelString, label, NULL); 
XtVaSetValues(save_indicator, XmNshadowType, XmSHADOW_OUT, NULL); 
psdl_modified = true; 
syntax_checked = false; 

} 

else { 

label = XmStringCreateSimple("”); 

XtVaSetValues(save_indicator, XmNlabelString, label, NULL); 
XtVaSetValues(save_indicator, XmNshadowType, XmSHADOW_IN, NULL); 

} 

XmStringFree(label); 
error label (); 


void update_status(char ^status, BOOLEAN bell) { 

XtVaSetValues(status_indicator, XmNvalue, status, NULL); 
if (bell) 

XBell(display_ptr,100) ; 

} 


void clear_status() { 

XtVaSetValues (status_indicator, XmNvalue, NULL); 

} 


// Initializes the color table. 

void initialize_color_table(Screen ^screen) { 

Colormap color__map = DefaultColormapOfScreen (screen) ; 

XColor color, unused; 

int i, screen_depth = DefaultDepthOfScreen(screen); 

if (screen_depth > 1) { //a color screen 

for(i =1; i <= MAXCOLORS; i++) { 

if ( ! XAllocNamedColor (display_ptr, color__map, 
colors[i - 1], &color, Sunused)) 

printf ("Allocated unknown color: %s\n", colors[i-1]); 
color_table[i] = color.pixel; 

} 

} 

else { // a black and white screen 

for(i =1; i <= MAXCOLORS; i++) { 

if (Strcmp(colors[i - 1], "White") != 0) 

color__table [i] = BlackPixelOfScreen (screen) ; 
else 

color_table [i] = WhitePixelOfScreen(screen); 

} 

} 

} 


*•*•★•*• 
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* Executes menu options from the *file* menu. This is 

* called by either the menu callback function, if the 
pulldown menus are used, or by the draw() function, 

* if the alt-key combinations are used. 

•k -k ! 


void handle_file_options(int item_no) { 
int action; 


Quest_Script abort_script = 

{"","Abort changes made to graph?", "Yes", 

BTN2}; 

Quest_Script save_script = 

{"","Save changes made to graph?", "Yes", 

BTNl}; 


No", 

"Cancel 

No", 

"Cancel 


XFlush(display_ptr); 
switch(item_no) { 

case 0: // Save 

// MY 8/5/97 

if { still_open() ) 

{ 

warning(drawing_a, "Please close other windows first"); 
break; 

} 

// Check for error condition of no Root...this should not be 
possible 

next_action_ptr->option = SAVE_TO_DISK; 

next_action_ptr“>reinvoke = true; 

free(next_action_ptr->next_op); 

next_action_ptr“>next_op = graphic_list^current_op_name{); 

next_action_ptr“>next_op_num = graphic_list.current_op_num{); 
return_sde_flag = true; 

break; 

case 1: // Restore from Save 

// MY 8/5/97 

if { still_open() ) 

{ 

warning(drawing_a, "Please close other windows first"); 
break; 

} 

// Check for error condition of no Root...this should not be 
possible 

action = YES; // Default action if not 

modified 

if (psdl_modified) 

action = AskUser(app,drawing_a, abort_script); 
switch(action) { 
case YES: 

next_action_ptr->option = REVERT; 

next_action_ptr“>reinvoke = true; 

free(next_action_ptr“>next_op); 
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next_action_ptr->next_op = graphic_list.current_op_name() 

next_action_ptr->next_op_num = graphic_list.current_op_num(); 
return_sde_flag = true; ~ 

break; 

case NO: 

return_sde_flag = false; // Aborted operation, do nothing 
break; 

} 

break; 

case 2: // Print 

// MY 8/5/97 

if ( still_open() ) 

{ 

warning(drawing_a, "Please close other windows first"); 
break; 

} 

AskPrint(app,drawing_a, &PrintCmd); 
if (PrintCmd.answer == OK) { 

XSendEvent(display_ptr, toplevel_window. True, 0, print_event); 

} 


break; 


/* 

* MY 
*/ 

case 3: // Abandon Changes 

// MY 8/5/97 

if ( still_open() ) 

{ 

warning(drawing_a, "Please close other windows first"); 
break; 

) 


Quest_Script abandon_script = 

{"","A11 changes will be lost, are you sure?", 

"Yes","No","Cancel", BTNl}; 

action = AskUser(app, drawing_a, abandon_script); 
if (action == YES) { 

next_action_ptr->option = ABANDON; 

next_action_ptr->reinvoke = true; 

free(next_action_ptr->next_op); 

next_action_ptr->next_op = graphic_list.current_op_name(); 

next_action_ptr->next_op_num = graphic_list.current_op_num(); 
return_sde_flag = true; 

} 

break; 

case 4: // Exit 

// MY 8/5/97 

if { still_open() ) 

{ 
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warning (clrawing_a, "Please close other windows first”); 
break; 


} 


action = NO; // Default action if not modified 

// This is not the default save option, see 

save_script 

if (psdl_modified) 

action = AskUser(app,drawing_a, save_script); 

switch{action) { 
case YES: 

next_action_ptr~>option = SAVE_TO_DISK; 
next_action_ptr->reinvoke = false; 
free{next_action_ptr->next_op); 

next_action_ptr->next_op = graphic_list. root_op_name () ; 

next_action_ptr->next_op_num = graphic_list. root_op_num {) ; 
return_sde_flag = true; 
break; 

case NO: 

next_action_ptr->option = ABANDON; 

next_action_ptr->reinvoke = false; 

free(next_action_ptr->next_op); 

next_action_ptr->next_op = graphic_list.root_op_name{); 

next_action_ptr->next_op_num = graphic_list.root_op_num() ; 
return_sde_flag = true; 

break; 

case CANCEL: 
default: 

return_sde_flag = false; 
break; 

} 

break; 
default: 
break; 

} 

} 

* Executes menu options from the ^psdl* menu. This is 

* called by either the menu callback function, if the 

* pulldown menus are used, or by the draw{) function, 

* if the alt-key combinations are used. 

'k'k'k'k'k'icir'k'kif-k'k'k'k'k'k-k-ieir-ic-ic'k'k'k'k'k’k'k'k'k'k-^c'k'k'k’k'k’k-ic-ic'k'k'k'k'k'k'k'k'k'k'k'k'k'k’k’k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k'k 

void handle_psdl_options{int item_no) { 

int action; 
char *opName; 

Quest_Script abort_script = 

{"”,"Abort changes made to graph?", "Yes", "No", "Cancel", 

BTN2}; 
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Quest_Script save_script = 

{"","Save changes made to graph?", "Yes", "No", "Cancel", 

BTNl}; 

XFlush(display_ptr); 
switch{item no) { 


/* 

* MY 
*/ 

case 

// 

if 

{ 


} 


3: // Syntax Check 

MY 8/5/97 
( still_open() ) 

warning(drawing_a, "Please close other windows first" 
break; 


next_action_ptr->option = CHECK_SYNTAX; 

next_action_ptr“>reinvoke = true; 

free(next_action_ptr“>next_op); 

next_action_ptr->next_op = graphic_list.current_op_name(); 

next_action_ptr->next_op_num = graphic_list.current_op_num(); 
return_sde_flag = true; 


break; 


/* 

* unmasked 8/6/97 
*/ 

case 0: //Go to Root 

// MY 8/5/97 

if ( still_open() ) 

{ 

warning(drawing_a, "Please close other windows first"); 
break; 

} 

// Check for error condition of no Root...this should not be 
possible 

if (graphic_list.root_op_num() == UNDEFINED_OPNUM) { 
warning(drawing_a,"No Root node defined"); 

break; 

} 


next_action_ptr->option = UPDATE_TREE; 

next_action_ptr->reinvoke = true; 

free(next_action_ptr“>next_op); 

next_action_ptr“>next_op = graphic_list.root_op_name(); 

next__action_ptr“>next_op_num = graphic_list. root_op_num () ; 
return_sde_flag = true; 

break; 


case 1: //Go to Parent 


// MY 8/5/97 
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if ( still_open() ) 

{ 

warning (cirawing_a, "Please close other windows first"); 
break; 

} 

// Check for error condition of no Parent 
if (graphic_list.parent_op_num() == UNDEFINED_OPNUM) { 
warning(drawing_a,"No parent node defined"); 

break; 

} 

next_action_ptr->option = UPDATE_TREE; 

next_action_ptr->reinvoke = true; 

free(next_action_ptr->next_op); 

next_action_ptr->next_op = graphic_list.parent_op_name(); 

next_action_ptr->next_op_num = graphic_list.parent_op_num(); 

return_sde__f lag = true; 

break; 


case 2: // Decompose 

// MY 8/5/97 

if ( still_open() ) 

{ 

warning(drawing_a, "Please close other windows first"); 
break; 

} 


if (selected_object_ptr == NULL) 

warning(drawing_a,"Please select an operator"); 


else { 

if (selected_object_ptr->is_a() == OPERATOROBJECT) { 

opName = selected_object_ptr->name() ; 

if (strchr(opName,’.*) != NULL) { // Is a type 

warning(drawing_a,"Not allowed to decompose a Type Operator"); 
update_status( 

"A Type Operator must be Atomic: rename or leave Atomic", 
RING_BELL); 
free(opName); 

} 


else { 

next_action_ptr->option = UPDATE_TREE; 

next_action_ptr->reinvoke = true; 
free (next__action_ptr->next_op) ; 
next_action_ptr->next_op = opName; 

next_action_ptr->next_op_num = 

((OperatorObject *) selected_object_ptr)- 


>op_num(); 

return_sde_flag = true; 

} 

} 


else 

warning(drawing_a,"Please select an operator"); 

} 


break; 
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default: 

return_sde_flag = false; 
break; 

} 

} 


// This function is called when a selection is made from 
// the list box displayed in the *draw_options:Color' menu. 

void color_list_cb(Widget widget, XtPointer, 

XtPointer cb_struct_ptr) { 
XmListCallbackStruct *list_struct_ptr = 

(XmListCallbackStruct *) cb_structj>tr; 

if (selected_object_ptr != NULL) { 

if (selected_object_ptr->is_a() == OPERATOROBJECT) { 
selected_object_ptr->erase() ; 

selected_object_ptr->color (list_struct_ptr->item_position) ; 
selected_obj ect_ptr->draw(SOLID); 
save_state(SAVE_REQUIRED); 

} 

} 

else 

default_color = list_struct_ptr->item_position; 
XtDestroyWidget(widget); 


// This function is called when a selection is made from 
// the list box displayed in the ’draw_options;Font’ menu. 

void font_list_cb(Widget widget, XtPointer, 

XtPointer cb_struct_ptr) { 
XmListCallbackStruct *list_struct_ptr = 
(XmListCallbackStruct *) cb_struct_ptr; 

if (selected_object_ptr != NULL-) { 
selected_object_ptr->erase(); 

selected_obj ect_ptr->set_obj ect_font(list_struct_ptr- 
>item_position); 

selected_object_ptr->draw(SOLID); 
save_state(SAVE^REQUIRED); 

} 

else { 

default^font = list_struct_ptr->item_position; 
graphic_list.set_default_font(default_font); 

} 

XtDestroyWidget(widget); 

XmProcessTraversal (drawing__a, XmTRAVERSE_CURRENT) ; 


// This function is called when a selection is made from 
// the list box displayed in the 'draw_options:Undelete Operator’ 
// menu. 

static void op_list_cb(Widget widget, XtPointer, 

XtPointer cb_struct_ptr) { 

XmListCallbackStruct *list_struct_ptr = 

(XmListCallbackStruct *) cb_struct_ptr; 
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// The last entry in the list is ’Cancel*. 

if (list_struct_ptr“>item_position != num_del_ops +1) { 

graphic_list.set_undeleted(OPERATOROBJECT, 

del_op_id[list_struct_ptr->item_position 

- 1 ]); 

save_state(SAVE_REQUIRED) ; 
graphic_list.draw(); 

} 

XtDestroyWidget(widget); 

XmProcessTraversal(drawing_a, XmTRAVERSE_CURRENT)/ 


// Executes menu options from the ’Edit* menu. This is 
// called by either the menu callback function, if the 
// pulldown menus are used, or by the draw() function, 

// if the alt-key combinations are used. 

void handle_edit_options(int item_no) { 
int if num_items = XtNumber(colors) ; 
int reply; 

XmStringTable color^list, font_list, op_list; 

Widget list_box, op_box; 

char *del_op_str[MAXDELETEDOPS] ; 

switch(item_no) { 
case 0: 

// MY 8/5/97 

if ( still_open() ) 

{ 

warning (drawing_a, ’’Please close other windows first”); 
break; 

} 


color^list = 

(XmStringTable) XtMalloc(num_iterns * sizeof(XmString *)); 
for(i =0; i < num_iterns; i++) 

color_list [i] = XmStringCreateSimple(colors[i]); 
list_box = 

XmCreateScrolledList (drawing_a, ’’Colors”, NULL, 0) ; 
XtVaSetValues(list_box, 

XmNiterns, color_list, 

XmNitemCount, num_items, 

XmNvisibleltemCount, 8, 

NULL); 

for(i =0; i < num_items; i++) 

XmStringFree(color_list[i] ) ; 

XtFree((char *) color_list) ; 

XtAddCallback(list_box, XmNdefaultActionCallback, 
color_list_cb, NULL); 

XtManageChild(list_box) ; 
break; 

case 1: 

// MY 8/5/97 

if ( still_open() ) 

{ 

warning (drawing^a, ’’Please close other windows first”); 
break; 
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} 


font_list = 

(XmStringTable) XtMaHoc(MAXFONTS * sizeof(XmString *)) 
for(i =0; i < MAXFONTS; i++) 
font_list[i] = 

XmStringCreateSimple (graphic_list. font_name (i + 1) ) 
list_box = 

XmCreateScrolledList(drawing_a, "Fonts", NULL, 0) 
XtVaSetValues(list_box, 

XmNitems, font_list, 

XmNitemCount, MAXFONTS, 

XmNvisibleltemCount, 7, 

NULL); 

for(i =0; i < MAXFONTS; i++) 

XmStringFree{font_list[i]) ; 

XtFree{{char *) font_list); 

XtAddCallback(list_box, XmNdefaultActionCallback, 
font_list_cb, NULL); 

XtManageChild(list_box) ; 
break; 

case 2: 

// MY 8/5/97 

if { still_open{) ) 

{ 

warning(drawing_a, "Please close other windows first"); 
break; 


if (selected_objecHptr != NULL) { 
selected_object_ptr->unselect() ; 
selected_object_ptr = NULL; 

} 

graphic_list. get__del_op_list (del_op_str, del_op_id, 

nuin_del_ops) ; 

op_list = (XmStringTable) 

XtMalloc((num_del_ops +1) * sizeof(XmString *)) 
for(i =0; i < num_del_ops; i++) 

op_list[i] = XmStringCreateSimple(del_op_str[i]); 
op_list[num_del_ops] = XmStringCreateSimple("Cancel"); 
op__box = XmCreateScrolledList (drawing_a, "Undelete", 

NULL, 0); 

XtVaSetValues(op_box, 

XmNitems, op_list, 

XmNitemCount, num_del_ops + 1, 
XmNvisibleltemCount, 7, 

NULL); 

for(i =0; i < num_del_ops + 1; i++) 

XmStringFree(op_list[i]) ; 

XtFree((char *) op^list) ; 

XtAddCallback(op_box, XmNdefaultActionCallback, 
op_list_cb, NULL); 

XtManageChild(op_box); 
break; 

/★ 

* MY 

* 

case 3: 
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// MY 8/5/97 

if ( still_open() ) 

{ 

warning(drawing_a/ "Please close other windows first"); 
break; 

} 


Quest_Script abandon_script = 

{""^"All changes will be lost, are you sure?", 

"Yes","No","Cancel", BTNl}; 

reply = AskUser(app, drawing_a, abandon_script); 
if (reply == YES) { 

next_action_ptr->option = ABANDON; 

next_action__ptr->reinvoke = true; 
free(next_action_ptr->next_op); 

next_action_ptr->next_op = graphic_list.current_op_name() 

next_action_ptr->next_op_num = graphic_list.current_op_num(); 
return_sde_flag = true; 

} 

break; 


V 


case 3: 

XFillRectangle(display_ptr, drawing_area_pixmap, 
erase_context, 0, 0, width, height); 
XFillRectangle(display_ptr, draw_window, 

erase_context, 0, 0, width, height); 
graphic_list.draw(); 

break; 

default: 
break; 

} 

XmProcessTraversal (drawing^a, XmTRAVERSE^CURRENT) ; 


void handle_layout_options(int item_no) { 
int i, X, y, node_count; 
int to_id, from_id, icount; 

GraphObject *temp_ptr, *next_ptr; 

OperatorObject *operators[500]; 

OperatorObject *obj_ptr; 

StreamObject *str_ptr; 

SplineObject *spl_ptr, spline; 
graph G; 

node* V = new node[500]; 

// BUILD LEDA GRAPH FROM CAPS GRAPH 
node_count = 0; 

temp_ptr = (GraphObject *) graphic_list.cur_graph(); 
G.clear(); 

while (temp_ptr != NULL) 

{ 

if (temp_ptr“>is_a() == OPERATOROBJECT) 

{ 

obj_ptr = (OperatorObject *) temp_ptr; 
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V[node_count] = G.new_node(); 
operators[node_count] = obj_ptr; 
obj_ptr->set_location(x,y); 
obj_ptr->set_default_text_location() ; 

graphic_list.move_notify(obj_ptr->is_a() , obj_ptr->id()); 

cout « 

node_count++; 

) 

else if (temp_ptr->is_a() == STREAMOBJECT) 

{ 

str_ptr = (StreamObject *) temp_ptr; 
to_id = -9; 
from_id = -9; 

for (icount=0; icount<node_count; icount++) 

{ 

if (str_ptr->to() == operators[icount]“>id()) 
to_id = icount; 

if (str_ptr->from() == operators[icount]~>id()) 
from_id = icount; 

) 

if ( (to_id >= 0) && (to_id < node_count) && 

(from_id >= 0) && (from_id < node_count) ) 

{ 

G.new_edge(V[from_id], V[to_id]); 
cout << 

} 

else cout << "ERROR PARSING GRAPH EDGE in handle_layout"; 


} 

else 

{ 

cout << "?"; 

} 

next_ptr = temp_ptr->next(); 
temp_ptr = next_ptr; 

} 

cout << endl; 


// RUN ALGORITHM 
switch (item_no) { 
case 0: 

cout << "Got 
cout << "Got 
run_ortho(G, 
break; 


orthoganal layout parameter" << endl; 
" « item_no << " parameter" « endl; 
TRUE); 


case 1: 


case 2: 


case 


cout « "Got 
cout << "Got 
run_straight_ 
break; 

cout << "Got 
cout << "Got 
run_straight^ 
break; 

cout « "Got 
cout « "Got 
run_tutte(G, 
break; 


straight line 2 layout parameter" << endl; 
" « item_no << " parameter" « endl; 
line2(G, TRUE); 


straight line layout parameter" « endl; 
" << item_no << " parameter" « endl; 
line(G, TRUE); 


tutte layout parameter" << endl; 

" << item_no << " parameter" « endl; 
TRUE); 
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cout << "Got D2 spring embedder layout parameter" << endl; 
cout << "Got " « item_no << " parameter" << endl; 
run_d2_spring_embedding(G, TRUE); 
break; 

case 5: 

cout << "Got spring embedder layout parameter" << endl; 
cout << "Got " « item_no << " parameter" << endl; 
run_spring_embedding(G, TRUE); 
break; 

default; 

cout << "Got undefined parameter" << endl; 
break; 


} 

// SET CAPS POSITIONS 

// DRAW GRAPH 
graphic_list.draw(); 

// graphic_list.draw(); 

// XmProcessTraversal (drawing_a, XmTRAVERSE_CURRENT) ; 
// warning(drawing_a,"Not yet implemented."); 


void handle_tool_options(int item_no) { 

warning(drawing_a,"Not yet implemented.") ; 


void set_color(Widget widget, char *color) { 

Display *dpy = XtDisplay(widget) ; 

Colormap cmap = DefaultColormapOfScreen(XtScreen(widget)) ; 
XColor col, unused; 

if ( !XAllocNamedColor(dpy, cmap, color, &col, &unused)) { 

warning(drawing_a,"Can’t allocate color"); 
return; 

} 

XSetForeground(dpy, std_graphics_context, col.pixel); 


* Menu call-back functions. These functions are called by the window 

* manager when a menu option is selected from a pull-down menu. The 

* item which was selected is passed in client_data. 


Static void file_menu_cb(Widget, XtPointer client_data, XtPointer) { 
int item_no = (int) client_data; 

handle_file_options(item_no); 


static void psdl_menu_cb(Widget, XtPointer client_data, XtPointer) { 
int item_no = (int) client_data; 

handle_psdl_options(item_no); 


static void edit_menu_cb (Widget, XtPointer client__data, XtPointer) { 
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int iteni_no = (int) client_data; 
handle_edit_options(item_no); 


static void layout_menu_cb(Widget, XtPointer client_data, XtPointer) { 
int item_no = (int) client_data; 

handle_layout_options(item_no); 


static void tool_menu_cb(Widget, XtPointer client_data, XtPointer) { 
int item_no = (int) client_data; 

handle_tool_options (itein_no) ; 


static void help__menu_cb (Widget w, XtPointer client_data, 

XtPointer call_data) { 
int itein_no = (int) client_data; 

help_cb(drawing_a, help_menu_files[item_no], call_data); 


void help_cb(Widget w, XtPointer client_data, XtPointer call_data) { 
// Implemented by Doug Lange 8/19/96 

Widget help_dialog, pane, text_w, rc, action_a; 
struct stat statb; 

char ch, *buf; 
int i = 0, n = 0; 
int len = 0; 

Arg args[10]; 

static ActionArealtem action_items[] = { 

{"OK", close_dialog, NULL} 

}; 


help_dialog = XtVaCreatePopupShell("Help", 

xmDialogShellWidgetClass, XtParent(w), 
XmNdeleteResponse, XmDESTROY, 

NULL); 

pane = XtVaCreateWidget("pane", xmPanedWindowWidgetClass, help_dialog, 

XmNsashWidth, 1, 

XmNsashHeight, 1, 

NULL); 

rc = XtVaCreateWidget("control_area", xmRowColumnWidgetClass, pane, 
NULL); 

stat((char*)client_data, &statb); 
ifstream from((char *)client_data); 
len = statb.st_size; 

buf = new char[len +1]; // Add a space for NULL 
i = 0; 

// while (from.get(ch) && (i < HELPSIZ -1)) { 

while (from.get(ch) && (i < len)) { 
buf[i] = ch; 
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i++ ; 

} 

buf[i] = (char)NULL; 


XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 


XmNscrollVertical, true); n++; 

XmNscrollHorizontal, false); n++; 

XmNeditMode, XmMULTI_LINE_EDIT) 

XmNeditable, false); n++; 

XmNcursorPositionVisible,false); n++; 
XmNwordWrap, true); n++; 

XmNvalue, buf); n++; 

XmNrows, 20); n++; 

XmNwidth, 525); n++; 


n+-i- 


text_w = XmCreateScrolledText(rc, ”help_text”, args, n) 


delete buf; 


XtManageChild(text_w); 

XtManageChild(rc); 

action_iterns[0].data = (XtPointer)help_dialog; 
action_a = CreateActionArea(pane, action_items, 
XtNumber(action_iterns)); 

XtManageChild(pane); 

XtPopup(help_dialog, XtGrabNone); 


} 


void build_inenu_bar (Widget &main_w. Widget &menubar) { 

// 8/4/96 KBM Updated for label changes in Req 4 

// Also changed callback names to reflect new labels. 


// ?? Need to look at short-cut keys.... not implemented correctly 


XmString 

file_menu, 

psdl_menu, 

edit_menu, 
refresh_opt, 

// tool_menu, 

layout_menu, 
d2_se_opt, 

help_menu, 
exception_opt, 


save_opt, restore_opt, print_opt, exit_opt, 
syntax_check_opt, goto_root_opt, goto_parent_opt, 
decompose_opt, 

color_opt, font_opt, undelete_opt, abandon_opt, 
reuse_lib_opt, 

ortho_opt, str_line2_opt, str_line_opt, tutte_opt, 
spring_opt, 

psdl_grammar_opt, operator_opt, stream_opt, 
timer_opt; 


Widget widget; 


file_menu 
save_opt 
restore_opt 
print_opt 
exit_opt 


= XmStringCreateSimple("File”); 

= XmStringCreateSimple("Save"); 

= XmStringCreateSimple("Restore from Save"); 
= XmStringCreateSimple("Print”); 

= XmStringCreateSimple("Exit”); 


psdl_menu = 

syntax_check_opt= 
goto_root_opt = 


XmStringCreateSimple("PSDL"); 
XmStringCreateSimple("Syntax Check"); 
XmStringCreateSimple("Go to Root"); 
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goto_parent_opt = XmStringCreateSimple("Go to Parent"); 
decompose_opt = XmStringCreateSimple("Decompose"); 


edit_menu = 

color_opt = 

font_opt = 

undelete_opt = 

abandon_opt = 

refresh_opt = 

/* 

tool_menu = 

reuse_lib_opt = 

printf("Just before 


XmStringCreateSimple("Edit"); 
XmStringCreateSimple("Color"); 
XmStringCreateSimple("Font") ; 
XmStringCreateSimple("Undelete Operator"); 
XmStringCreateSimple("Abandon Changes"); 
XmStringCreateSimple("Refresh Display"); 

XmStringCreateSimple("Tools") ; 
XmStringCreateSimple("Reuse Library"); 

layout menu.\n"); 


layout_menu 

ortho_opt 

str_line2_opt 

str_line_opt 

tutte_opt 

d2_se_opt 

spring_opt 


= XmStringCreateSimple("Layout"); 
XmStringCreateSimple("Orthogonal Layout"); 

= XmStringCreateSimple("Straight Line 2"); 

= XmStringCreateSimple("Straight Line"); 

= XmStringCreateSimple("Tutte Layout"); 

= XmStringCreateSimple("D2 Spring Embedding"); 
= XmStringCreateSimple("Spring Embedding"); 


help_menu = XmStringCreateSimple("Help"); 

psdl_grammar_opt= XmStringCreateSimple("PSDL Grammar") 
operator_opt = XmStringCreateSimple("Operators"); 

stream_opt = XmStringCreateSimple("Streams"); 

exception_opt = XmStringCreateSimple("Exceptions"); 


timer_opt 


= XmStringCreateSimple("Timers"); 


menubar = XmVaCreateSimpleMenuBar(main_w, "menubar", 
XmVaCASCADEBUTTON, file_menu, NULL, 

XmVaCASCADEBUTTON, psdl_menu, NULL, 

XmVaCASCADEBUTTON, edit_menu, NULL, 

// XmVaCASCADEBUTTON, tool_menu, NULL, 

XmVaCASCADEBUTTON, layout_menu, NULL, 

XmVaCASCADEBUTTON, help_menu, NULL, NULL); 

if (widget = XtNameToWidget(menubar, "button_4")) // Assign 

to help 

XtVaSetValues(menubar, XmNmenuHelpWidget, widget, NULL); 


XmVaCreateSimplePulldownMenu(menubar, "file_menu", 0, 


XmVaPUSHBUTTON, 

save opt. 

NULL, 

NULL, 

NULL, 

XmVaPUSHBUTTON, 

restore_opt, 

NULL, 

NULL, 

NULL 

XmVaPUSHBUTTON, 

print opt. 

NULL, 

NULL, 

NULL 

7/22/97 MY 

XmVaPUSHBUTTON, 

abandon_opt, 

NULL, 

NULL, 

NULL 


file menu cb. 


XmVaPUSHBUTTON, exit_opt, NULL, NULL, NULL, 

NULL); 


XmVaCreateSimplePulldownMenu(menubar, "psdl_menu", 1, psdl_menu_cb, 

/* 

* 7/22/97 MY 

* XmVaPUSHBUTTON, syntax_check_opt, NULL, NULL, NULL, 

*/ 

XmVaPUSHBUTTON, goto_root_opt, 'R', NULL, NULL, 
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XmVaPUSHBUTTON, goto_parent_opt, 'P't NULL, NULL, 
XmVaPUSHBUTTON, decompose_opt, 'D', NULL, NULL, 
NULL); 


XmVaCreateSimplePulldownMenu(menubar, "edit menu", 2, edit menu cb. 


XmVaPUSHBUTTON, color_opt, 
XmVaPUSHBUTTON, font_opt, 
XmVaPUSHBUTTON, undelete_opt, 

/* 

* 7/22/97 MY 

■Ar 

XmVaPUSHBUTTON, abandon_opt, 




XmVaPUSHBUTTON, refresh_opt, 
NULL); 


NULL, NULL, NULL, 
NULL, NULL, NULL, 
NULL, NULL, NULL, 


NULL, NULL, NULL, 
’f*, NULL, NULL, 


/★ 

XmVaCreateSimplePulldownMenu(menubar, "tool_menu", 3, tool_menu_cb, 
XmVaPUSHBUTTON, reuse_lib_opt, *U', NULL, NULL, 

NULL); 

XmVaCreateSimplePulldownMenu(menubar, "layout_menu", 3, 
layout_menu_cb, 

XmVaPUSHBUTTON, ortho_opt, NULL, NULL, NULL, 

XmVaPUSHBUTTON, str_line2_opt, NULL, NULL, NULL, 

XmVaPUSHBUTTON, str_line_opt, NULL, NULL, NULL, 

XmVaPUSHBUTTON, tutte_opt, NULL, NULL, NULL, 


XmVaPUSHBUTTON, 

d2 se opt. 

NULL, 

NULL, 

NULL, 

XmVaPUSHBUTTON, 
NULL); 

spring opt. 

NULL, 

NULL, 

NULL, 

XmVaCreateSimplePulldownMenu(menubar, "help menu", 4, 

XmVaPUSHBUTTON, 

psdl_grammar opt. 

NULL, 

NULL, 

NULL, 

XmVaPUSHBUTTON, 

operator__opt, 

NULL, 

NULL, 

NULL, 

XmVaPUSHBUTTON, 

stream_opt, 

NULL, 

NULL, 

NULL, 

XmVaPUSHBUTTON, 

exception opt. 

NULL, 

NULL, 

NULL, 

XmVaPUSHBUTTON, 
NULL); 

timer_opt. 

NULL, 

NULL, 

NULL, 


XmStringFree(file_menu); 

XmStringFree(save_opt) ; 
XmStringFree(restore_opt) ; 
XmStringFree(print_opt) ; 
XmStringFree(exit_opt) ; 
XmStringFree (psdl__menu) ; 

XmStringFree (syntax_check_opt) ; 
XmStringFree(goto_root_opt) ; 
XmStringFree(goto_parent_opt); 
XmStringFree(decompose_opt) ; 
XmStringFree(edit_menu) ; 
XmStringFree(color_opt) ; 
XmStringFree(font_opt); 
XmStringFree(undelete_opt) ; 
XmStringFree(abandon_opt) ; 
XmStringFree(refresh_opt) ; 
//XmStringFree(tool_menu) ; 

// XmStringFree(reuse_lib_opt); 
XmStringFree (layout__menu) ; 
XmStringFree(ortho_opt) ; 
XmStringFree(str_line2_opt); 
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XmStringFree(str_line_opt); 
XmStringFree(tutte_opt); 
XmStringFree(d2_se_opt) ; 
XmStringFree(spring_opt) ; 
XmStringFree(help_menu); 

XmStringFree(psdl_grammar_opt); 
XmStringFree(operator_opt); 
XmStringFree(stream_opt) ; 
XmStringFree(exception_opt); 
XmStringFree(timer_opt); 


// Creates the push buttons used to select the tools. 


void make_buttons(Widget &rowcol, 

Widget &op_button. 

Widget &term_button, 
Widget &stream_button. 

Widget &select_button. 
Widget &types_button, 
Widget &spec_button. 

Widget &timers_button/ 
Widget &informal_button, 


// tools 


// types 

// current op spec 

// current op impl 
// current op impl 


Pixmap &op_button_pixmap, 

Pixmap & t e rm_bu11 on_pixmap, 
Pixmap &stream_button_pixmap, 
Pixmap &select_button_pixmap, 
Pixmap &types_button_pixmap, 
Pixmap &spec_button_pixmap, 

Pixmap &timers_button_pixmap, 
Pixmap &informal_button_pixmap, 
Display *display_ptr, 

Screen *screen_ptr) { 


static Widget op_btn_bb, term_btn_bb, stream_btn_bb, select_btn_bb, 
types_btn_bb, spec_btn_bb, timers_btn_bb, 

informal btn bb; 


Window root_window = RootWindowOfScreen(screen_ptr)/ 
unsigned int screen_depth = DefaultDepthOfScreen(screen_ptr); 


op_button_pixmap = XCreatePixmap(display_ptr, root_window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 
term_button_pixmap = XCreatePixmap(display_ptr, root_window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 
stream_button_pixmap = XCreatePixmap(display_ptr, root_window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 
select_button_pixmap = XCreatePixmap(display_ptr, root_window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 
types_button_pixmap = XCreatePixmap(display_ptr, root_window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 
spec_button_pixmap = XCreatePixmap(display_ptr, root_window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 
timers_button_pixmap = XCreatePixmap(display_ptr, root^window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 
informal_button_pixmap = XCreatePixmap(display_ptr, root_window, 

BUTTONWIDTH-4, BUTTONWIDTH-4, screen_depth); 

XFillRectangle(display_ptr, (Drawable) op_button_pixmap. 
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erase_context, 0, 
XFillRectangle(display_ptr, 
erase_context, 0, 
XFillRectangle(display_ptr, 
erase_context, 0, 
XFillRectangle(display_ptr, 
erase_context, 0, 
XFillRectangle(display_ptr, 
erase_context, 0, 
XFillRectangle(display_ptr, 
erase_context, 0, 

XFillRectangle(display_ptr, 
erase_context, 0, 
XFillRectangle(display_ptr, 
erase context, 0, 


0, BUTTONWIDTH-4, BUTTONWIDTH-4); 

(Drawable) term_button_pixmap, 

0, BUTTONWIDTH-4, BUTTONWIDTH-4); 

(Drawable) stream_button_pixinap, 

0, BUTTONWIDTH-4, BUTTONWIDTH-4); 

(Drawable) select_button_pixmap, 

0, BUTTONWIDTH-4, BUTTONWIDTH-4); 

(Drawable) types_button_pixmap, 

0, BUTTONWIDTH-4, BUTTONWIDTH-4); 

(Drawable) spec_button_pixmap, 

0, BUTTONWIDTH-4, BUTTONWIDTH-4); 

(Drawable) tiniers_button_pixmap, 

0, BUTTONWIDTH-4, BUTTONWIDTH-4); 

(Drawable) informal_button_pixmap, 
0, BUTTONWIDTH-4, BUTTONWIDTH-4); 


XSetLineAttributes(display_ptr, std_graphics_context, 2, 

LineSolid, CapButt, JoinMiter); 


XDrawArc(display_ptr, (Drawable) op_button_pixmap, 
std_graphics_context, 

10, 15, BUTTONWIDTH-(3*10) , BUTTONWIDTH-(3*10) , 

CIRCLE_BEGIN, FULL_CIRCLE) ; 

XDrawRectangle (display_ptr, (Drawable) terni_button_pixmap, 
std_graphics_context, 

10, 15, BUTTONWIDTH-(3*10), BUTTONWIDTH-(3*10)); 

XDrawLine(display_ptr, (Drawable) stream_button_pixmap, 
std_graphics_context, 

10, 15, BUTTONWIDTH-(2*10), BUTTONWIDTH-(2*10) ) ; 

XDrawString(display_ptr, (Drawable) select_button_pixmap, 

std_graphics_context, 10, (BUTTONWIDTH/2)+5, "Select", 6) 
XDrawString(display_ptr, (Drawable) types_button_pixmap, 

std_graphics_context, 10, (BUTTONWIDTH/2)+5, "Types ", 6) 
XDrawString (display_ptr, (Drawable) spec_button_pixinap, 

std_graphics_context, 10, (BUTTONWIDTH/2)+5, " Spec ", 6) 
XDrawString (display_ptr, (Drawable) timers_button_pixinap, 

std_graphics_context, 10, (BUTTONWIDTH/2)+5, "Timers", 6) 
XDrawString(display_ptr, (Drawable) informal_button_pixmap, 

std_graphics_context, 10, (BUTTONWIDTH/2)-8, "Graph ", 6) 

XDrawString(display_ptr, (Drawable) informal_button_pixmap, 

std_graphics_context, 5, (BUTTONWIDTH/2)+5, "Informal", 

8) ; 

XDrawString(display_ptr, (Drawable) informal_button_pixmap, 

std_graphics_context, 10, (BUTTONWIDTH/2)+18, "Desc ", 

6) ; 


XmString button_label; 

button_label = XmStringCreateSimple("Operator") ; 

op_button = XtVaCreateManagedWidget("op_button", 
xmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 
XmNshadowType, XmSHADOW_OUT, 
XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTH, 
XmNlabelType, XmSTRING, 
XmNlabelString, button_label, 
//XmNlabelType, XmPIXMAP, 


264 



//XmNlabelPixmap, op_button_pixmap, 

NULL); 

XmStringFree(button_label) ; 

button_label = XmStringCreateSimple(" Term"); 

term_button = XtVaCreateManagedWidget("term_button", 
xmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 
XmNshadowType, XmSHADOW_OUT, 

XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTH, 

XmNlabelType, XmSTRING, 

XmNlabelString, button_label, 
//XmNlabelType, XmPIXMAP, 

//XmNlabelPixmap, op_button_pixmap, 

NULL); 

XmStringFree(button_label) ; 

button_label = XmStringCreateSimple(" Stream"); 

stream_button = XtVaCreateManagedWidget("stream_button", 
xmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 
XmNshadowType, XmSHADOW_OUT, 

XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTH, 

XmNlabelType, XmSTRING, 

XmNlabelString, button_label, 
//XmNlabelType, XmPIXMAP, 
//XmNlabelPixmap, op_button_pixmap, 

NULL); 

XmStringFree(button_label); 

button_label = XmStringCreateSimple(" Select"); 

select_button = XtVaCreateManagedWidget("select_button", 
XmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 

XmNshadowType, XmSHADOW_OUT, 

XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTH, 

XmNlabelType, XmSTRING, 

XmNlabelString, button_label, 

//XmNlabelType, XmPIXMAP, 

//XmNlabelPixmap, op_button_pixmap, 

NULL); 

XmStringFree(button_label); 

button_divider = XtVaCreateManagedWidget ( "separator", 

xmSeparatorWidgetClass, rowcol, 
// XmNy, R0W14 - 8, 

// XmNwidth, WIN_WIDTH, 

NULL); 
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button_label = XmStringCreateSimple(" Types"); 

types_button = XtVaCreateManagedWidget("types_button", 
xmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 

XmNshadowType, XmSHADOW_OUT, 

XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTK, 

XmNlabelType, XmSTRING, 

XmNlabelString, button_label, 

//XmNlabelType, XmPIXMAP, 

//XmNlabelPixmap, op_button_pixmap, 

NULL); 

XmStringFree(button_label); 

button__label = XmStringCreateLtoR( " Parent\n Spec", 

XmSTRING_DEFAULT_CHARSET) / 

spec_button = XtVaCreateManagedWidget("spec_button", 
xmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 

XmNshadowType, XmSHADOW_OUT, 

XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTH, 

XmNlabelType, XmSTRING, 

XmNlabelString, button_label, 

//XmNlabelType, XmPIXMAP, 

//XmNlabelPixmap, op_button_pixmap, 

NULL); 

XmStringFree(button_label) ; 

button_label = XmStringCreateSimple(" Timers"); 

timers_button = XtVaCreateManagedWidget("timers_button", 
XmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 

XmNshadowType, XmSHADOW_OUT, 

XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTH, 

XmNlabelType, XmSTRING, 

XmNlabelString, button_label, 

//XmNlabelType, XmPIXMAP, 

/ /XmNlabelPixmap, op_butt on__pixmap, 

NULL); 

XmStringFree(button_label); 

button_label = XmStringCreateLtoR(" Graph\n Desc", 

XmSTRING_DEFAULT_CHARSET); 

inf ormal__button = XtVaCreateManagedWidget ( "informal_button" , 
XmDrawnButtonWidgetClass, 
rowcol, 

XmNrecomputeSize, false. 
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XmNpushButtonEnabled, false, 

XmN s ha dowT yp e, XmS H ADOW_OUT, 

XmNwidth, BUTTONWIDTH, 

XmNheight, BUTTONWIDTH, 

XmNlabelType, XmSTRING, 

XmNlabelString, button_label, 
//XmNlabelType, XmPIXMAP, 
//XmNlabelPixmap, op_button_pixmap, 

NULL); 

XmStringFree(button_label); 

XSetLineAttributes(display_ptr, std_graphics_context, 1, 

LineSolid, CapButt, JoinMiter); 


// Redraws the drawing canvas. 

void redraw(Widget, XtPointer, 

XtPointer cbs) { 

XmDrawingAreaCallbackStruct '*^temp_ptr; 

temp_ptr = (XmDrawingAreaCallbackStruct *) cbs; 

XCopyArea (temp__ptr->event->xexpose . display, 

drawing_area_pixmap, temp_ptr->window, 
std_graphics_context, 0, 0, width, height, 0, 0); 


// Draws a square black box on the canvas to aid in 
// graphic manipulation of objects. 

void draw_handle(GC graphics_context, int x, int y) { 

X -= HANDLESIZE / 2; 
y -= HANDLESIZE / 2; 
if (X < 0) 

X = 0; 

if (y < 0) 

y = 0; 

// When the display function is set to GXxor, the pixel being 
// written is exclusive-or’ed with the target pixel to 
// determine color. This means that writing the same pixel with 
// the same color twice restores the original color, simplifying 
// the process of erasing handles. 

XSetFunction (.display_ptr, graphics_context, GXxor); 
XFillRectangle(display_ptr, draw_window, graphics_context, 

X, y, HANDLESIZE, HANDLESIZE); 

XFillRectangle(display_ptr, drawing_area_pixmap, 
graphics_context, 

X, y, HANDLESIZE, HANDLESIZE); 

XSetFunction(display_ptr, graphics_context, GXcopy); 

} 


// This function erases the temporary guidelines used when 
// streams are drawn. 

// Dotted lines are erased first, then handles. Since each 
// handle is overwritten with the following dotted line, an 
// erased handle makes an erased blotch in the beginning of the 
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// next segment. When the next segment is written in xor mode, 

// it makes a black mark where the erased handle overwrote the 
// beginning of its segment. 

void erase_guides(OP_ID from_stream_id, SplineObject *temp_spline_ptr) { 

OperatorObject *temp_operator_ptr; 

XYPAIR line__start, line_end; 

temp_spline_ptr->reset_iter(); 
if (from_stream_id 1= UNDEFINED_OPNUiyi) { 
temp_operator_ptr = (OperatorObject *) 

graphic_list.target_object(OPERATOROBJECT, from_stream_id); 
line__start = temp_operator_ptr->center () ; 

} 

else 

line_start = temp_spline_ptr->next_pair(); 
line_end = temp_spline_ptr->next_pair(); 
while(line_end.x != -1) { 

XDrawLine(display_ptr, draw_window, dotted_context, 
line_start.X, line_start.y, line_end.x, 
line_end.y); 

XDrawLine (display_ptr, drawing_area_pixmap, dotted_context, 
line_start.X, line_start.y, line_end.x, 
line_end.y); 
line_start = line_end; 

line_end = temp_spline_ptr->next_pair() ; 

} 

temp_spline_ptr->reset_iter() ; 
line_end = temp_spline_ptr->next_pair(); 
while(line_end.X != -1) { 

draw_handle(std_graphics_context, line_end.x, line_end.y); 
line_end = temp_spline_ptr->next_pair() ; 

} 


// This function is called when a stream is being drawn 
// and the mouse is clicked on either a clear spot on the 
// drawing canvas, or on top of another stream. If a double- 
// click is registered, the user wants to terminate an external 
// stream. 


void handle_null_point(OP_ID from_stream_id, int &last_point_x, 

int &last_point_y, 

int &x_state, int &y_state, 

XEvent in_event, 

SplineObject *temp_spline_ptr, BOOLEAN &done, 
GraphObject *&temp_object_ptr, 

StreamObject *&temp_stream_ptr) { 

// Checks for two clicks in the same spot. 


if ((from_stream_id != UNDEFINED_OPNUM) && 

((last_point_x - (HANDLESIZE / 2) - HITFUDGE) 

< in_event.xbutton.x) && 

((last_point_x + (HANDLESIZE / 2) + HITFUDGE) 

> in_event.xbutton.x) && 

((last_point_y - (HANDLESIZE / 2) - HITFUDGE) 

< in^event.xbutton.y) && 

((last_point_y + (HANDLESIZE / 2) + HITFUDGE) 

> in event.xbutton.y)) { 
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erase_guides (from_stream_id, temp_spline_ptr) ; 

OP_ID new_id = graphic_list.request_id(STREAMOBJECT); 

// MY 8/4/97 

sprintf(default_name, "noname_%d”, get_unique_id()); 

//MY: ”” -> default_name 

temp_stream_ptr = new StreamObject (default_name, new_id, 

froin_stream_id, 

0, UNDEFINED_TIME, MS, 
temp_spline_ptr,// @6 
true, false); 

temp_stream_ptr->set_object_ptrs (&graphic_list) ; 

graphic_list.add(temp_stream_ptr) ; 

save_state(SAVE_REQUIRED); 

temp_stream_ptr“>draw(SOLID) ; 

temp_stream_ptr = NULL; 

temp_object_ptr = NULL; 

done = true; 

temp_spline_ptr->clear(); 

} 

else { 

x_state = in_event.xbutton.x; 
y_state = in_event.xbutton.y; 
temp_spline_ptr->add(x_state, y_state) ; 

XDrawLine(display_ptr, draw_window, dotted_context, 

last_point_x, last_point_y, x_state, y_state); 
XDrawLine(display_ptr, drawing_area_pixmap, dotted_context, 
last_point_x, last_point_y, x_state, y_state); 
draw_handle (std_graphics_context, x_state, y__state) ; 

#ifdef GE_DEBUG 

// cout << "ge: " << x_state << " " << y_state << " ” << 

// HANDLESIZE « endl; 

#endif 

last_point_x = x_state; 
last_point_y = y_state; 

} 

} 


// Once the user selects the Stream Tool and begins to draw, 

// the draw_stream () function handles all events to speed up 
// performance. 

void draw_stream(int initial_x, int initial_y) { 

GraphObject *temp_object_ptr; 

OperatorObject *conv_ptr; 

XYPAIR temp_pair; 

char buffer[INPUT_LINE_SIZE]; // added for req #6.4 dha 
int count =0; // added for req #6.4 dha 

int bufsize = INPUT_LINE_SIZE; // added for req #6.4 dha 
OP_ID from_stream_id; 

int x_state, y_state, last_point_x, last_point_y; 
unsigned long stream_event_mask = 

(ButtonPressMask | PointerMotionMask | 
KeyPressMask); 

unsigned long normal_mask = (ButtonPressMask | PointerMotionMask | 

KeyPressMask | 

ButtonMotionMask | ExposureMask | 
ButtonReleaseMask); 

XEvent in_event; 

StreamObject *temp_stream_ptr; 
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SplineObject *temp_spline_ptr; 

BOOLEAN done = false; 

KeySym keysym; // added for req #6.4 dha 
temp_spline_ptr = new SplineObject; 

temp_object_ptr = graphic_list.hit(initial_x, initially); 
if (temp_object_ptr == NULL) { // External stream 

from_stream_id = UNDEFINED_OPNUM; 
temp_spline_ptr->add(initial_x, initial_y); 
x_state = initial_x; 
y_state = initial_y; 

draw_handle(std_graphics_context, x_state, y_state); 

} 

else { 

if (temp_object_ptr->is_a() != OPERATOROBJECT) { 

// External Stream 

from_stream_id = UNDEFINED_OPNUM; 
temp_spline_ptr->add(initial_x, initial_y); 
x_state = initial_x; 
y_state = initial_y; 

draw_handle(std_graphics_context, x_state, y_state); 

} 

else { 

conv_ptr = (OperatorObject *) temp_object_ptr; 

from_stream_id = conv_ptr->id(); 

temp_object_ptr = NULL; 

temp_pair = conv_ptr->center() ; 

x_state = temp_pair.x; 

y_state = temp_pair.y; 

} 

} 

last_point_x = x_state; 
last_point_y = y_state; 

XSelectInput(display_ptr, draw_window, 
stream_event_mask) ; 

while(done == false) { // monitors the event loop 

XNextEvent(display_ptr, &in_event); 
if (in_event.xbutton.window ~ draw_window) { 
switch(in_event.type) { 
case MotionNotify: 

#ifdef GE_DEBUG 

// cout << "Motion” << endl; 

#endif 


XDrawLine(display_ptr, draw_window, dotted_context, 

last_point_x, last_point_y, x_state, y_state) 
XDrawLine(display_ptr, drawing_area_pixmap, 

dotted_context, last_point_x, last_point_y, 
x_state, y_state); 
x_state = in_event.xbutton.x; 
y_state = in_event.xbutton.y; 

XDrawLine(display_ptr, draw_window, dotted_context, 

last_point_x, last_point_y, x_state, y_state) 
XDrawLine(display_ptr, drawing_area_pixmap, 

dotted_context, last__point_x, last_point_y, 
x_state, y_state); 

break; 

case ButtonPress: 
case KeyPress: 
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#ifdef GE_DEBUG 

if (in_event.type == ButtonPress) { 
// cout << ’’buttonpress” << endl; 

} 

else { 

// cout << ’’keypress” << endl; 

} 

#endif 


XDrawLine(display^ptr, draw_window, dotted_context, 

last_point_x, last_point_y, x_state, y_state); 
XDrawLine(display_ptr, drawing_area_pixmap, 

dotted_context, last_point_x, last_point y, 
x_state, y_state); 

temp_object_ptr = graphic_list,hit(in^event.xbutton.x, 

in_event.xbutton.y); 

if (temp_object_ptr == NULL) { 

handle_null_point(from_stream_id, last_point_x, 

last_point_y, x_state, y_state, 
in_event, temp_spline_ptr, done, 
temp_object_ptr, temp_stream_ptr); 

} 

else 

if (temp_object_ptr->is_a0 == OPERATOROBJECT) { 
erase_guides(from_stream_id, temp_spline_ptr); 

OP_ID new_id = graphic_list.request_id(STREAMOBJECT) 
// MY 8/4/97 


sprintf (default_name, ”noname_%d’*, get_unique_id()); 
tempostream_ptr = 

//MY; ”” -> default_name 

new StreamObject(default_name, new_id, 


from stream id, 


temp_object_ptr->id() , 


UNDEFINED_TIME, MS, // @6 

temp_spline_ptr, true, false) 
temp_stream_ptr->set_object_ptrs(&graphic_list); 
save_state(SAVE_REQUIRED); 

graphic_list.add(temp_stream_ptr); 
temp_stream_ptr->draw(SOLID) ; 
temp_stream_ptr = NULL; 
temp_object_ptr = NULL; 
done = true; 

temp_spline_ptr->clear(); 


} 

else 

if (temp_object_ptr->is_a() == STREAMOBJECT) { 
handle__null_point (f rom_stream_id, last_point_x, 
last_point_y, x_state, y_state, 
in_event, temp_spline_ptr, done, 
temp_object_ptr, temp_stream_ptr); 


} 


if (in_event.type == KeyPress) { 

count = XLookupString(&in_event.xkey, buffer, 
bufsize, &keysym, NULL); 
buffer[count] = NULL; /* add NULL terminator */ 


if (keysym == XK_Escape) { 
//temp_stream_ptr->erase(); 

//OP_ID deleted_op_id = temp_stream_ptr->id{); 
//graphic_list.delete_notify(temp_stream_ptr-> 
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// is_a{), deleted_op_id); 

//temp_stream_ptr->set__deleted () ; 

//temp_stream_ptr = NULL; 
graphic_list.draw(); 


done = true; 

} 

else { 

XBell{display_ptr, 100); 

} 


break; 

default: 

/ 

break; 

} //switch 
} //if right window 
} //while done == false 
done = false; 

XSelectInput{display_ptr, draw_window, normal_mask) ; 


// Draws the outline of the text being moved. 

void draw_text_shadow (int x, int y, int width, int height) { 

XDrawRectangle {display_ptr, drawing_area_pixmap, 

dotted_context, x - width / 2, y - height / 2, 
width, height); 

XDrawRectangle{display_ptr, draw_window, dotted_context, 

X - width / 2, y - height / 2, width, height); 


// The main draw routine. This function is called by the 
// window manager every time the mouse is moved, a mouse button 
// pressed, or a key pressed inside the draw window. It is 
// called with a string token that indicates why it was called, 

// and processes the event accordingly. 

void draw(Widget, XEvent *event. String *args. Cardinal *) { // void 

draw 

static char string[INPUT_LINE_SIZE]; // added for req #6.1.1 dha 
static OperatorObject *temp_operator_ptr = NULL; 
static StreamObject *temp_stream__ptr = NULL; 
static BOOLEAN first_draw = true, handle_selected = false, 

text_selected = false, drawing_changed = false; 
static int x_state, y_state, shadow_height, shadow_width; 
static OP_ID from_stream_id; 

GraphObject *temp_object_ptr = NULL; 
static GraphObject *ibar_object_ptr = NULL; 
char *warningMSG; 

char buffer[INPUT_LINE_SIZE]; // added for req #6.1.1 dha 

int count =0; // added for req #6.1.1 dha 

int length =0; // added for req #6.1.1 dha 

int bufsize = INPUT_LINE_SIZE; // added for req #6.1.1 dha 

int X = event->xbutton.x; 

int y = event->xbutton.y; 

char *labelName; 
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OperatorObject *conv_op_ptr; 

StreamObject *conv_st_ptr; 

KeySym keysym; // added for req #6.1.1 dha 
BOOLEAN state_change, type_match; 

BOOLEAN type_operator; 

if (strcmp(args[0], "down”) == 0) { // Button pressed 

clear_status (); 

XmProcessTraversal (drawing_a, XmTRAVERSE_CURRENT) ; 
x_state = x; 
y_state = y; 

if (tool_state == SELECT_TOOL) { 
if (selected_object_ptr != NULL) { 

if (selected_object_ptr->hit_handle(x, y)) { 

handle_selected = true; 

object_def = selected_object_ptr->is_a(); 
if (object_def == OPERATOROBJECT) { 

op_being_updated = (OperatorObject *)selected_object_ptr; 
delete temp_operator_ptr; 

conv_op_ptr = (OperatorObject *) selected_object_ptr; 
temp_operator_ptr = 

new OperatorObject("", UNDEFINED_OPNUM, UNDEFINED_OPNUM, 
UNDEFINED_TIiyiE, MS, conv_op_ptr->x () , // @6 

conv_op_ptr->y(), 
conv_op_ptr->radius() , 
default_color, false, 
conv_op_ptr“>is_composite(), 
conv_op_ptr->is_terminator()); 
temp_operator_ptr->set_handle_selected( 

conv_op_ptr->handle_selected()); 

} else if (object_def == STREAMOBJECT) { 

st_being_updated = (StreamObject *)selected_object_ptr; 

} 

} // selected_object_ptr->hit_handle() 

else { // Unselects previously selected object 

handle_selected = false; 
selected_object_ptr->unselect(); 
selected_object_ptr = NULL; 
delete temp_operator_ptr; 
temp_operator_ptr = NULL; 

} 

} // selected_object_ptr != NULL 

if (handle_selected == false) { 

temp_object_ptr = graphic_list,hit(x, y) ; 
if (temp_object_ptr != NULL) { 
temp_object_ptr->select() ; 
selected_object_ptr = temp_object_ptr; 

text_selected = selected_object_ptr->text_selected(); 
object_def = temp_object_ptr“>is_a() ; 
if (object_def == OPERATOROBJECT) { 

op_being_updated = (OperatorObject *) temp_object_ptr; 

// Makes temporary operator to move around 
delete temp_operator_ptr; 

conv_op_ptr = (OperatorObject *) temp_object_ptr; 
temp_operator_ptr = 

new OperatorObject("", UNDEFINED_OPNUM, UNDEFINED_OPNUM, 
UNDEFINED_TIME, MS, conv_op_ptr->x(), // @6 

conv_op_ptr->y(), 
conv_op_ptr“>radius(), 
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default_color, false, 
conv_op_ptr“>is_coinposite () , 
conv_op_ptr->is_terminator()); 

} else if (object_def == STREAMOBJECT) { 

st_being_updated = (StreamObject *) temp_object_ptr; 

} 

} // temp_object_ptr != NULL 
else { // No object selected 
teinp_object_ptr = NULL; 
selected_object_ptr = NULL; 
delete teinp_operator_ptr; 
temp_operator_ptr = NULL; 


} // handle_selected == false 
} /// tool_state == SELECT_TOOL 

else { // button down, operator tool selected? 

// if ((((tool_state == OPERATOR_TOOL) || 

// (tool_state == TERMINATOR_TOOL)) 

// (ibar_itiode != true) ) | | 

8/22/96 dha, 

// (object_def != OPERATOROBJECT)) { 

6.3 


ScEc 


// added 
// req. 6.2 & 


if ((tool_state == OPERATOR_TOOL) 1| 

(tool_state == TERMINATOR_TOOL)) { 

OP_ID new_id = graphic_list.request_id(OPERATOROBJECT); 

OP_ID new_op = graphic_list.request_id(OPERATOROBJECT); 
if (tool_state == OPERATOR_TOOL) { 

// MY 8/4/97 

sprintf(default_name, "noname_%d", get_unique_id()); 
teinp_operator_ptr = 

// BROCKETT 1/22/93 default x and y values changed from 0 to 

100 

new OperatorObject(default_name, new_id, new_op, //MY: "" -> 

default_name 

UNDEFINED_TIME, MS, 100, 100, 30, // @6 

default_color, true, false, 
false); 

temp_operator_ptr->set_location(x, y); 

} // tool_sate == OPERATOR_TOOL 
else 

if (tool_state == TERMINATOR_TOOL) { 

// MY 8/4/97 

sprintf(default_name, "noname_%d", get_unique_id()); 
temp_operator__ptr = 

// BROCKETT 1/22/93 default x and y values changed from 0 to 

100 


new OperatorObject (default_name, new__id, new_op, //MY: "" 


“> default_p 


UNDEFINED_TIME, MS, 100, 100, 30, // @6 

default_color, true, false, 
true); 

temp_operator_ptr->set_location(x, y); 

} // tool_state == TERMINATOR_TOOL 
graphic_list.add((GraphObject *) temp_operator_ptr); 
save_state(SAVE_REQUIRED); 

temp_operator_ptr->draw(SOLID); 
temp_operator_ptr = NULL; 

} // tool state == OPERATOR TOOL || TERMINATOR TOOL && ibar mode 


else // button down, stream tool selected? 
if (tool state == STREAM TOOL) { 
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draw_stream(x, y); 

} 

} // button down, operator tool selected? 

} else if (strcmp(args[0], "motion”) == 0) { // button not down 

if (tool_state == SELECT_TOOL) { 
if (selected_object_ptr != NULL) { 
drawing_changed = true; 
if (text_selected) { 

if (first_draw == true) { 

shadow_width = selected_object_ptr->text_width(); 
shadow_height = selected_object_ptr->text__height(); 
draw_text_shadow(x, y, shadow_width, shadow_height); 
first_draw = false; 

} // first_draw 
else { 

draw_text_shadow (x_state, y_state, shadow_width, 
shadow_height); 

draw_text_shadow(x, y, shadow_width, shadow_height); 

} 

} // text_selected 
else 

if (selected_object_ptr->is_a() == OPERATOROBJECT) { 
if (handle_selected == true) { 

if (first_draw == true) { 
selected_object_ptr->erase(); 
selected_object_ptr->unselect(); 
selected_object_ptr->draw(SOLID); 
temp_operator_ptr->draw(DOTTED); 
first_draw = false; 

} // first_draw 
else { 

temp_operator_ptr->move_handle(x - x_state, 

y “ y_state); 

} 

} // handle_selected 
else { 

if (first_draw == true) 

// Drawing the same thing twice in xor mode erases it. When 
// moving an object, it is drawn once the first and last time, 

// and twice afterwards 

first_draw = false; 
else 

temp_operator_ptr->draw(DOTTED); 

temp_operator_ptr->move(x - x_state, y - y_state); 
temp_operator_ptr->draw(DOTTED); 

} 

} // is_a OPERATAOROBJECT 
else { 

if (selected_object_ptr->is_a() == STREAMOBJECT) { 
if (handle_selected) { 
if (first_draw == true) { 
conv_st_ptr = 

(StreamObject *) selected_object_ptr; 
conv_st_ptr“>erase_handle(); 
draw_handle(std_graphics_context, x, y) ; 
first_draw = false; 

} // first_draw 
else { 
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draw_handle(std_graphics_context, x_state, y_state) 
draw_handle(std_graphics_context, x, y); 
selected_object_ptr->inove_handle (x - x_state. 


} // handle^selected 
} // is_a STREAMOBJECT 

} 

x_state = x; 
y_state = y; 

} // selected_object_ptr 1= NULL 
} // tool_state == SELECT_TOOL 

// I-bar mode check 

temp_object_ptr = graphic_list.over(x, y) ; 
if (temp_object_ptr != NULL) { 

object_def = temp_object_ptr->is_a() ; 
if (object_def == OPERATOROBJECT) { 

#ifdef GE_DEBUG 

// cerr << "It is an Operator Object" « endl; 

#endif GE_DEBUG */ 

ibar_mode = true; 

setcursor(drawing_a, True, XC_xterm) ; 

} // object_def == OPERATOROBJECT 
else 

if (object_def == STREAMOBJECT) { 

#ifdef GE_DEBUG 

// cerr << "It is an Stream Object" « endl; 

#endif GE_DEBUG */ 

ibar_mode = true; 

setcursor(drawing_a, True, XC_xterm); 

} // object_def == STREAMOBJECT 
else { 

ibar_mode = false; 

setcursor(drawing_a. False, None); 

} 

} // temp_object_ptr 1= NULL 
else { // No object selected 

#ifdef GE_DEBUG 

// cerr « "No object selected Object" « endl; 

#endif GE_DEBUG */ 

ibar_mode = false; 

setcursor(drawing_a. False, None); 

} // No object selected 
} else if (strcmp(args[0], "up") == 0) { 

if (tool^state == SELECT_TOOL) 

if (selected_object_ptr != NULL) { 
if (text_selected) { 

if (first_draw == false) { 

draw_text_shadow(x_state, y_state, 

shadow__width, shadow_height) ; 
selected_object_ptr->text_locate(x, y); 
save_state(SAVE_REQUIRED) ; 

} // first draw 
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} // text_selected 
else 

if (selected_object_ptr->is_a() == OPERATOROBJECT) { 

if (first_draw == false) { 

temp_operator_ptr->draw(DOTTED); 

XYPAIR temp_pair = teinp_operator_ptr->center () ; 
conv_op_ptr = 

(OperatorObject *) selected_object_ptr; 
conv_op_ptr-‘>radius (temp_operator_ptr->radius () ) ; 
conv_op_ptr->set_location(temp_pair.x, 

temp_pair.y); 
if (handle_selected) 

conv_op_ptr->set_default_text_location(); 

} // first_draw 
} // is_a OPERATOROBJECT 
else 

if ((selected_object_ptr->is_a() == STREAMOBJECT) 

&& (handle_selected)) { 

draw_handle(std_graphics_context, x_state, y_state); 

} // is_a STREAMOBJECT ~ 

if (drawing_changed == true) { 

graphic_list .move_notif y (selected_object_ptr->is_a () , 

selected_obj ect_ptr->id () ) ; 
graphic_list.draw() / 
save_state (SAVE_REQUIRED) ; 
drawing_changed = false; 

} // drawing_changed 
handle_selected = false; 

} // selected_object_ptr != NULL 
first_draw = true; 

} else if (strcmp(args[0], "btnSdown”) == 0) { 

clear_status(); 
if (ibar_mode == true) { 

if (object_def == OPERATOROBJECT) { 

// op_being_updated = (OperatorObject *) temp_object_ptr; 
operator_property_dialog (drawing_a, op_being_updated, x, y, 

graphic_list. cur_op_is_terminator () , 
graphic_list.avail_impl_langs_adr(), 
&graphic_list); 

} 

else 

if (object_def == STREAMOBJECT) { 

stream_property_dialog (drawing_a, st_being_updated, x, y, 
&graphic_list); 

// XFlush(XtDisplay(drawing_a)); /* Stub for stream code */ 

} 

else { 

XFlush(XtDisplay(drawing_a));/* Stub for Non Operator of Stream 

* / 

} 

} // ibar_mode 

} else if (strcmp(args[0], "btnSmotion”) == 0) { 

temp_object_ptr = graphic_list.over(x, y) ; 
if (temp_object_ptr != NULL) { 

object_def = temp_object_ptr->is_a(); 
if (object_def == OPERATOROBJECT) { 

#ifdef GE_DEBUG 

// cerr << "It is an Operator Object" « endl; 

#endif /* GE DEBUG */ 
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ibar_mode = true; 

setcursor(drawing_a, True, XC_xterm) ; 

op_being_updated = (OperatorObject *) temp_object_ptr; 

} 

else 

if (object_def =- STREAMOBJECT) { 

#ifdef GE_DEBUG 

// cerr << "It is an Stream Object” « endl; 

#endif GE_DEBUG */ 

ibar_mode = true; 

setcursor(drawing_a, True, XC_xterm); 
st_being_updated = (StreamObject *) temp_object_ptr; 


} 

else { 

ibar_mode = false; 

setcursor(drawing_a. True, XC_left_ptr)/ 

} 

} 

else { // No object selected 
#ifdef GE_DEBUG 

cerr << "No object selected Object" << endl; 

#endif /* GE_DEBUG */ 

ibar_mode = false; 

setcursor(drawing_a. False, None) ; 

} 

} else if (strcmp(args[0], "btnSup") == 0) { 

XFlush(XtDisplay (drawing_a)); /* Stub for stream code */ 

} else if (strcmp(args[0], "motionnotify") == 0) { 

if (label_edit_mode =- true) { 
label_edit_mode = false; 
if (ibar_object_ptr) { 

if (ibar_object_ptr-'>is_a 0 == STREAMOBJECT) { 

labelName = ((StreamObject *)ibar_object_ptr)->name(); 

// MY 

if ( is_empty (labelName) ) 

{ 

sprintf(default_name, "noname_%d", get_unique_id()); 
labelName = default_name; 

((StreamObject *)ibar_object_ptr)->name(labelName); 

((StreamObject *)ibar_object_ptr)->draw(SOLID); 

} 


warningMSG = (char *) malloc(strlen(labelName)+40); 
if (!valid_id(labelName)) { 

sprintf(warningMSG,"Invalid stream name: %s", labelName); 
warning(drawing_a, warningMSG); 
update__status ( 

"Illegal stream name, retype: id ::= letter 
{alpha_numeric}", 

RING_BELL); 

((StreamObject *)ibar_object_ptr)->erase_text(); 

// MY 
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sprintf(default_name, "noname_%d", get_unique_id()); 

((StreamObject *)ibar_object_ptr)~>name(default_name); 

//((StreamObject *)ibar_object_ptr)->name(""); 

((StreamObject *)ibar_object_ptr)“>draw_text(SOLID) ; 

} else if (is_keyword(labelName, false)) { 

sprintf(warningMSG,"Stream name is a keyword: %s", labelName) 
warning(drawing_a, warningMSG); 

update_status("Stream name is a keyword, retype", RING_BELL); 
((StreamObject *)ibar_object_ptr)->erase_text(); 

// MY 

sprintf(default_name, ”noname_%d", get_unique_id()); 

((StreamObject *)ibar_object_ptr)->name(default_name) ; 

// ( (StreamOb j ect * ) ibar_obj ect_ptr) -*>name ( " " ) ; 

((StreamObject *)ibar_object_ptr)->draw_text(SOLID) ; 

} else { 

// Valid stream name, get any existing type information 
type_match = graphic_list.fetch_matching_stream_type( 
(StreamObject *)ibar_object_ptr, &state_change); 
if (state_change) 

((StreamObject *)ibar_object_ptr)“>draw(SOLID); 

} 

free(labelName); 
free(warningMSG); 

} 

else { 

labelName = ((OperatorObject *)ibar_object_ptr)->name(); 
type_operator = (strchr(labelName, * . * ) ) ? true : false; 

// MY 

if ( is_empty(labelName) ) 

{ 

sprintf(default_name, "noname_%d", get_unique_id()); 
labelName = default_name; 

((OperatorObject *)ibar_object_ptr)->name(labelName); 

( (OperatorObj ect *) ibar_object_ptr)-‘>draw (SOLID) ; 

} 




{ 


warningMSG = (char *) malloc(strlen(labelName)+80); 

if (!valid_op_id(labelName)) { 

sprintf(warningMSG, 

"Invalid operator name (syntax or keyword): %s", labelName) 
warning(drawing_a, warningMSG); 

update_status("Illegal operator name, retype:" 

" op_id ::= [id *.*] op_name [*(* [id_list] *r [id_list] 

RING_BELL); 

((OperatorObject * )ibar_object_ptr)->erase_text(); 

// MY 

sprintf(default_name, "noname_%d", get_unique_id()); 

((OperatorObject *)ibar_object_ptr)->name(default_name); 

//((OperatorObject *)ibar_object_ptr)->name(""); 

( (OperatorObject *) ibar_object_ptr)-*>draw_text (SOLID) ; 

} else if (type_operator && 

(((OperatorObject *)ibar_object_ptr)->is_composite())) 
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sprintf(warningMSG, 

"A Composite Operator can not be a Type: %s”, labelName) 
warning(drawing_a, warningMSG); 

update_status("Composite Operator can not be a Type:" 

" rename operator or make Automic", 

RING_BELL); 

((OperatorObject *)ibar_object_ptr)->erase_text (); 

//MY 

sprintf(default_name, "noname_%d", get_unique_id ()); 

((OperatorObject *)ibar_object_ptr)->name(default_name); 

//((OperatorObject *)ibar_object_ptr)->name(""); 

((OperatorObject *)ibar_object_ptr)->draw_text(SOLID); 

} else if (!type_operator && 

!graphic_list.unique_op_id(labelName, 

((OperatorObject *)ibar_object_ptr)->id())) { 

sprintf(warningMSG, 

"Simple Operator Names must be unique to level: %s", 

labelName); 

warning(drawing_a, warningMSG); 

update_status ("Operators that are not types must have a " 
"unique name", 

RING_BELL) ; 

((OperatorObject *)ibar_object_ptr)->erase_text(); 

// MY 

sprintf(default_name, "noname_%d", get_unique_id()); 

((OperatorObject *)ibar_object_ptr)->name(default_name); 
//((OperatorObject *)ibar_object_ptr)->name(""); 

((OperatorObject *)ibar_object_ptr)->draw_text(SOLID); 

} 

free(labelName); 

} 

} 

string[0] = NULL; 
buffer[0] = NULL; 
ibar_object_ptr = NULL; 

} 

temp_object_ptr = graphic_list.over(x, y) ; 
if (temp_object_ptr 1= NULL) { 

object_def = temp_object_ptr->is_a() ; 
if (object_def == OPERATOROBJECT) { 

#ifdef GE_DEBUG 

// cerr << "It is an Operator Object" << endl; 

#endif /* GE_DEBUG */ 

ibar_mode = true; 

setcursor(drawing_a. True, XC_xterm); 

op_being_updated = (OperatorObject *) temp_object_ptr; 

} 

else 

if (object_def == STREAMOBJECT) { 

#ifdef GE_DEBUG 

// cerr << "It is an Stream Object" « endl; 

#endif /* GE DEBUG */ 
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ibar_mode = true; 

setcursor(drawing_a, True, XC_xterm); 
st_being_updated = (StreamObj ect *) teinp_object_ptr; 

} 

else { 

ibar_mode = false; 

setcursor(drawing_a, True, XC_left_ptr); 

} 

} 

else { // No object selected 
#ifdef GE_DEBUG 

// cerr << ”No object selected Object” << endl; 

#endif /* GE_DEBUG */ 

ibar_mode = false; 

setcursor(drawing_a. False, None); 

} 

} else if (strcmp(args[0], ”key”) == 0) { 

#ifdef GE_DEBUG 

// cout << "key pressed: ” « event->xkey.keycode << endl 

tendif 

count = XLookupString(&event->xkey, buffer, 
bufsize, &keysym, NULL); 
buffer[count] = NULL; /* add NULL terminator */ 

if (label_edit_mode==true) { 

if ((keysym == XK_Return) || (keysym == XK_KP_Enter) || 

(keysym == XK_Linefeed)) { 

label_edit_mode = false; 
if (ibar_object_ptr) { 

if (ibar_object_ptr->is_a0 == STREAMOBJECT) { 

labelName = ( (StreamObject * ) ibar__obj ect_ptr)->name () ; 
warningMSG = (char *) malloc(strlen(labelName)+40); 
if (!valid_id(labelName)) { 

sprintf(warningMSG,"Invalid stream name: %s”, labelName) 
warning(drawing_a, warningMSG); 
update_status( 

"Illegal stream name, retype: id ::= letter 
{alpha_numeric}", 

RING_BELL); 

((StreamObject *)ibar_object_ptr)->erase_text (); 

((StreamObject *)ibar_object_ptr)->name("”); 

((StreamObject *)ibar_object_ptr)->draw_text(SOLID); 

} else if (is_keyword(labelName, false)) { 

sprintf(warningMSG,"Stream name is a keyword: %s”, 

labelName); 

warning(drawing_a, warningMSG); 

update_status("Stream name is a keyword, retype", 

RING_BELL); 

((StreamObject *)ibar_object_ptr)->erase_text(); 

( (StreamObj ect * ) ibar_object__ptr)->name (" ") ; 

((StreamObject *)ibar_object_ptr)->draw_text(SOLID); 

} else { 

// Valid stream name, get any existing type information 
type_match = graphic_list.fetch_matching_stream_type( 
(StreamObject *)ibar_object_ptr, &state_change); 
if (statexchange) 
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((StreamObject *)ibar_object_ptr)->draw(SOLID); 

} 

free(labelName); 
free(warningMSG); 

} 

else { 

labelName = ((OperatorObject *)ibar_object_ptr)->name(); 
type_operator = (strchr(labelName, *.*) ) ? true : false; 

warningMSG = (char *) malloc(strlen(labelName)+80); 
if (!valid_op_id(labelName)) { 

sprintf(warningMSG, 

"Invalid operator name (syntax or keyword): %s", 

labelName); 

warning(drawing_a, warningMSG); 

update_status("Illegal operator name, retype: " 

"op_id ::= [id *.*] op_name [* (* [id_list] *| * [id_list] 

RING_BELL) ; 

((OperatorObject *)ibar_object_ptr)->erase_text(); 

((OperatorObject *)ibar_object_ptr)->name(""); 

((OperatorObject *)ibar_object_ptr)->draw_text(SOLID); 

} else if ((strchr(labelName,*.’) !=NULL) && 

(((OperatorObj ect *)ibar_obj ect_ptr)- 

>is_composite())) { 

sprintf(warningMSG, 

"A Composite Operator can not be a Type: %s", 

labelName); 

warning(drawing_a, warningMSG); 

update_status("Composite Operator can not be a Type:" 

" rename operator or make Automic", 

RING^BELL); 

((OperatorObject *)ibar_object_ptr)->erase_text(); 

((OperatorObject *)ibar_object_ptr)->name(""); 

((OperatorObject *)ibar_object_ptr)->draw_text(SOLID); 

} else if (!type_operator && 

!graphic_list.unique_op_id(labelName, 

((OperatorObject *)ibar_object_ptr)->id())) { 

sprintf(warningMSG, 

"Simple Operator Names must be unique to level: %s", 
labelName); 

warning(drawing_a, warningMSG); 

update_status("Operators that are not types must have a " 
"unique name", 

RING_BELL) ; 

((OperatorObject *)ibar_object_ptr)->erase_text(); 

((OperatorObject *)ibar_object_ptr)->name(""); 

((OperatorObject *)ibar_object_ptr)->draw_text(SOLID); 

} 

free (labelName); 

) 

} 

string[0] = NULL; 
buffer[0] = NULL; 
ibar_object_ptr = NULL; 

} 

else 

if (((keysym >= XK_KP_Space) && (keysym <= XK_KP_9)) || 

((keysym >= XK_space) && (keysym <= XK_asciitilde))) { 

if ((strlen(string) + strlen(buffer)) >= INPUT_LINE_SIZE) { 
XBell(display_ptr, 100) ; 
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else { 

street(string, buffer); 

} 

} 

else 

if ((keysym >= XK_Shift_L) && (keysym <= XK_Hyper_R)) { 

; /* Do nothing because it *s a modifier key 

} 

else 

if ((keysym >= XK_F1) && (keysym <= XK_F35)) { 

if (buffer[0] != (char)NULL) { 

if ((strlen(string) + strlen(buffer)) >= INPUT_LINE_SIZE) { 
XBell(display_ptr, 100); 

} 

else { 

street(string, buffer); 

} 

} 

} 

else 

if .( (keysym == XK_BackSpace) | | 

(keysym == XK_Delete)) { 

if ((length = strlen(string)) > 0) { 

string[length - 1] = NULL; 

} 

else { 

XBell(display_ptr, 100); 

} 

} 


temp_object_ptr = graphic_list.over(x, y); 
if (temp_object_ptr != NULL) { 
object_def = temp_object_ptr->is_a() ; 
if (label_edit_mode != false && 

(object_def -= OPERATOROBJECT || 
object_def == STREAMOBJECT)) { 


// 


ibar_object_ptr = temp_object_ptr; 
temp_object_ptr“>erase_text(); 
temp__object__ptr->name (string) ; 
temp_object_ptr“>draw_text (SOLID) ; 
save_state (SAVE_REQUIRED) ; 
temp_object_ptr“>unselect(); 

} 

} 


} 

else 

if (alt_selected I| ctrl_selected) 
alt_selected = false; 
ctrl_selected = false; 
switch(keysym) { 
case XK_D: 
case XK_d: 

handle_psdl_options(2) ; 
break; 
case XK_P: 
case XK_p: 

handle_psdl_options(1) ; 
break; 
case XK R: 


// alt key pressed 


// Decompose 
// MY: 3 -> 2 
// Goto Parent 
// MY: 2 “> 1 


// Goto Root 
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// MY: 1 -> 0 


case XK_r: 

handle_psdl_options(0) / 
break; 

case XK_F; // Refresh Display 

case XK_f: 

handle_edit_options(4) ; 
break; 

case XK_Meta_L: // Alt key to activate 

case XK_Meta_R: 

alt_selected = true; 
break; 

case XK_Control_L: // Control key to activate 

case XK_Control_R: 

ctrl_selected = true; 
break; 
default: 
break; 

} 

} 

else if (keysym == XK_Meta_L 1 | keysyiu == XK_Meta_R) { 
alt_selected = true; 

} 

else if (keysym == XK_Control_L || keysym == XK_Control_R) { 
ctrl_selected = true; 

} 

else 

if (selected_object_ptr != NULL) { 
if ((keysym == XK_BackSpace) || 

(keysym == XK_Delete)) { 

selected_obj ect_ptr->erase(); 

OP_ID deleted_op_id = selected_object_ptr->id (); 

save_state(SAVE_REQUIRED); 
graphic_list.delete_notify(selected_obj ect_ptr-> 

is_a(), deleted_op_id); 
selected_obj ect_ptr“>set_deleted(); 
selected_object_ptr = NULL; 
graphic_list.draw(); 
ibar_mode = false; 

setcursor(drawing_a, False, None); 

} 

} 

else 

if (ibar_mode==true && 

label_edit_mode==false && 

( ( (keysym >= XK_KP_Space) && (keysym <= XK_KP__9) ) | | 

((keysym >= XK_space) && (keysym <= XK_asciitilde)))) { 

if ((strlen(string) + strlen(buffer)) >= INPUT_LINE_SIZE) { 
XBell(display_ptr, 100); 

} 

else { 

strcat(string, buffer); 

} 

label_edit_mode = true; 
clear_status() ; 

temp_object_ptr = graphic_list.over(x, y); 
if (temp_object_ptr != NULL) { 

object_def = temp_object_ptr->is_a(); 
if (object_def == OPERATOROBJECT || 
object_def == STREAMOBJECT) { 
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// temp_object_ptr~>select{); 

ibar_object_ptr = temp_object_ptr; 
temp_object_ptr->erase_text() ; 
temp_object_ptr->name(string); 
temp_object_ptr->draw_text(SOLID); 
save_state(SAVE_REQUIRED); 

// temp_object_ptr->unselect()/ 

} // label_edit_mode != false && () 

} // temp_object_ptr != NULL 
} // ibar_mode && labelled!t_mode == false && () 
} // strcmp KEY 

} // draw 


// Callback function. Just destroys the widget, 
void widget_killer(Widget widget, XtPointer, XtPointer) { 
XtDestroyWidget(widget); 

} 

// Callback function. Called when Operator Tool button is 
// pressed. 

void op_button_cb(Widget, XtPointer, XtPointer) { 
select_state(OPERATOR_TOOL); 

XmProcessTraversal(drawing_a, XmTRAVERSE_CURRENT)/ 
if (selected_object_ptr != NULL) { 
selected_object_ptr->unselect() ; 
selected_object_ptr = NULL; 

} 

//?? XtVaSetValues(tool_indicator, XmNvalue, "Operator Tool", NULL) 


// Callback function. Called when Terminator Tool button is 
// pressed. 

void term__button_cb (Widget, XtPointer, XtPointer) { 
select_state(TERMINATOR^TOOL)/ 

XmProcessTraversal(drawing_a, XmTRAVERSE_CURRENT); 
if (selected_object_ptr != NULL) { 
selected_object_ptr->unselect() ; 
selected_object_ptr = NULL; 

} 

//?? XtVaSetValues(tool_indicator, XmNvalue, "Terminator Tool", 
NULL); 

} 


// Null Callback. 

void null_cb(Widget, XtPointer, XtPointer) {} 


// Callback function. Called when Timer Tool OK button is 
// pressed. DL 8/22/96; KBM 10/24/96 
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void timer_tool_ok_cb(Widget parent, XtPointer client_data, 

XtPointer call data) { 


Widget 

XmAnyCallbackStruct 

int 

XmString 
char 
ID_LIST 
ID_LIST 
ID LIST 


list_w = (Widget)client_data; 

*cbs = (XmAnyCallbackStruct *)call_data; 
u_bound; 

*strlist; 

*text; 
op, tp; 

otimer = graphic_list,timer_list (); // MY 
idp, timers; 


enter timer = 0; // MY 8/5/97 


XtVaGetValues (list__w, 

XmNitemCount, &u_bound, 
XmNitems, &strlist, 
NULL); 


timers = NULL; 
if (u_bound >0) { 

idp = (ID_LIST) malloc(sizeof(ID_NODE) ) ; 
idp->next = NULL; 

//if (XmStringGetLtoR(strlist[0] , XmFONTLIST_DEFAULT_TAG, &text)) 

//@1 

if (XmStringGetLtoR(strlist[0], XmSTRING_DEFAULT_CHARSET, &text)) 

//@1 

idp->id = text; 
timers = idp; 

for (int i = 1/ i < u_bound; i++) { 

idp->next = (ID_LIST) malloc(sizeof(ID_NODE)); 
idp = idp->next; 
idp->next = NULL; 

//if (XmStringGetLtoR(strlist[i], XmFONTLIST_DEFAULT_TAG, 

&text))//@1 

if (XmStringGetLtoR(strlist[i], XmSTRING_DEFAULT_CHARSET, 

&text))//@ 1 

idp->id = text; 

} 

} 

// MY 

op = otimer; tp = timers; 

while ( op != NULL && tp != NULL ) 

{ 

if ( strcmp(op->id, tp->id) != 0 ) { 

save_state(SAVE_REQUIRED); 
break; 

} 

op = op->next; tp = tp->next; 

} 

if ( op != NULL ) save_state(SAVE_REQUIRED) ; 
if ( tp != NULL ) save_state(SAVE_REQUIRED) ; 

graphic_list.timer_list(timers); 
id_list_release(timers); timers = NULL; 

XtDestroyWidget(XtParent(XtParent(XtParent(parent)))); 
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} 


// Callback function. Called when Timer Tool button is 

// pressed. DL 8/16/96. 


void timers_button_cb(Widget parent,XtPointer client_data,XtPointer 
call data){ 


Widget dialog, rc, pane, list, action_a; 
int count =0, i, n=0; 

ID_LIST idp, timers; 

Arg args[5]/ 

XmString *str,string; 

static ActionArealtem action items[] = { 


}; 


{"OK", 

timer_tool ok cb. 

NULL}, 

{"Cancel”, 

timer_close_dialog. 

NULL}, 

{"Add”, 

timer tool add cb. 

NULL}, 

{"Delete”, 

t ime r_t ool_de l_cb, 

NULL}, 

{"Edit”, 

timer tool edit cb. 

NULL}, 

{"Help”, 

help_cb. 

"timers 


// MY 8/5/97 

if ( enter_timer == 1 ) { 

putchar(007); 
return; 

} 

if ( enter_errs == 1 ) { 

putchar(007); 

warning(parent, ^Please close error message window"); 
return; 

} 

■enter_timer = 1; 

//Build list for list widget 
timers = graphic_list.timer_list(); 

idp = timers; 
while(idp) { 
count++; 

idp = idp->next; 

} 

idp = timers; 

str = (XmString *) XtMalloc (count * sizeof (XmString)); 
for (i = 0; i < count; i++) { 

// str[i] = XmStringCreateLocalized(idp->id); // @1 

str[i] = XmStringCreateSimple(idp->id); // @1 

idp = idp“>next; 

} 

id_list_release(timers); timers = NULL; 

dialog = XtVaCreatePopupShell ( "dialog”, xmDialogShellWidgetClass, 

XtParent(parent), XmNtitle, "Timers Tool”, 
XmNdeleteResponse, XmDESTROY, 

NULL); 


action_items[1].data = (XtPointer)dialog; //Set cancel buttons 
client data 
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pane = XtVaCreateWidget("pane”, xmPanedWindowWidgetClass, dialog, 

XmNsashWidth, 1, 

XmNsashHeight, 1 , 

NULL); 

rc = XtVaCreateWidget ( "control_area", xmRowColumnWidgetClass, pane, 
NULL); 

// string = XmStringCreateLocalized("Enter or Edit Timers"); // @1 
string = XmStringCreateSimple("Enter or Edit Timers"); // @1 

XtVaCreateManagedWidget("label", xmLabelGadgetClass, rc, 

XmNlabelString, string, 

NULL); 

XmStringFree(string); 

list = XmCreateScrolledList(rc, "Timer_List", NULL, 0); 

XtVaSetValues(list, 

XmNvisibleltemCount, 10, 

XmNitemCount, count, 

XmNitems, str, 

NULL); 

XtManageChild(list) ; 
for(i = 0; i < count; i++) 

XmStringFree(str[i] ) ; 

XtManageChild(rc) ; 

//Set client data for "OK", "Add", "Del", and "Edit" buttons 

action_items[0].data = (XtPointer)list; 

action^items[2].data = (XtPointer)list; 

action_items[3].data = (XtPointer)list; 

action_items[4].data = (XtPointer)list; 

action_a = CreateActionArea(pane, action_items, 

XtNumber(action_iterns)); 

XtManageChild(pane); 

XtPopup(dialog, XtGrabNone); 


} 


// Callback function. Called when Informal Description Tool OK is 

// pressed. Added by Doug Lange 8/19/96. 

static void inform_tool_ok_pushed(Widget w, XtPointer client_data, 

XtPointer call_data) { 

Widget text_w = (Widget)client_data; 

XmAnyCallbackStruct *cbs = (XmAnyCallbackStruct *)call_data; 

char *text = XmTextGetString(text_w); 

char *org_text = graphic_list. graph_inf ormal__desc () ; 

enter_inform = 0; // MY 8/5/97 

// MY 

if (strcmp(text, org_text) != 0) { 

graphic_list. graph_informal_desc (text) ; 
save_state (SAVE_REQUIRED) ; 

} 

free(text); 
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XtDestroyWidget(XtParent(XtParent(XtParent(w) ) )) ; 


clear_status(); 

} 


// Callback function. Called when Informal Description Tool Button 

is 

// pressed. Added by Doug Lange 8/19/96. 


static void informal_button_cb(Widget w, XtPointer client data, 

XtPointer call_data) { 

Widget dialog, pane, rc, text_w, action_a; 

XmString string; 
char ^description; 


static ActionArealtem action_items 
{"OK”, inform_tool_ok_pushed, 

{"Cancel", inform_close_dialog, 
("Help", help^cb. 


[] = { 

NULL 

NULL 

"inform_tool.hip" 




// MY 8/5/97 

if ( enter_inform == 1 ) { 

putchar(007); 
return; 

} 

if ( enter_errs == 1 ) { 

putchar(007); 

warning(w, "Please close error message window"); 
return; 

} 

enter_inform = 1; 

dialog = XtVaCreatePopupShell ("dialog", xmDialogShellWidgetClass, 

XtParent(w), 

XmNtitle, "Informal Design Description", 
XmNdeleteResponse, XmDESTROY, 

NULL); 

action_items[1].data = (XtPointer)dialog; //Set cancel buttons 
client_data 

pane = XtVaCreateWidget("pane", xmPanedWindowWidgetClass, dialog, 

XmNsashWidth, 1, 

XmNsashHeight, 1, 

NULL); 

rc = XtVaCreateWidget("control_area”, xmRowColumnWidgetClass, pane, 
NULL); 

string = XmStringCreateSimple("Enter or Edit Informal Description"); 

//@1 

XtVaCreateManagedWidget("label", xmLabelGadgetClass, rc, 

XmNlabelString, string, 

NULL); 

XmStringFree(string); 

description = graphic^list.graph_informal_desc(); 
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int n = 0; 

Arg args[10] 

XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 
XtSetArg(args[n] 


XmNrows, 

XmNcolumns, 

XmNscrollVertical, 
XmNscrollHorizontal, 
XmNeditMode, 

XmNeditable, 

XmNcursorPositionVisible, 

XmNwordWrap, 

XmNvalue, 


12); n++; 

70); n++; 
true); n++; 
false); n++; 
XmMULTI_LINE_ 
true); n++; 
true); n++; 
true); n++; 
description); 


EDIT); n++ 


n++; 


text_w = XmCreateScrolledText(rc, "text-field", args, n) 
XtManageChild(text_w); 

//text_w = XtVaCreateManagedWidget("text-field", 
xmTextFieldWidgetClass, 

// rc, NULL)/ 


XtAddCallback(text_w, XmNmodifyVerifyCallback, validate_text, NULL) 
XtManageChild(rc); 

//Set client data for the "OK" and "Cancel" buttons 
action^items[0].data = (XtPointer)text_w; 

action_a = CreateActionArea(pane, action_items, 

XtNumber (action_iterns) ) ; 

//XtAddCallback(text_w, XmNactivateCallback, activate_cb, action_a) 

XtManageChild(pane); 
free(description); 

XtPopup(dialog, XtGrabNone); 


// Callback function. Called when Stream Tool button is 
// pressed. 

void stream_button_cb(Widget, XtPointer, XtPointer) { 
select_state(STREAM_TOOL); 

XmProcessTraversal(drawing_a, XmTRAVERSE_CURRENT); 
if (selected_object_ptr != NULL) { 
selected_object_ptr->unselect (); 
selected_object_ptr = NULL; 

} 

//?? XtVaSetValues(tool_indicator, XmNvalue, "Stream Tool", NULL); 


// Callback function. Called when Select Tool button is 
// pressed. 

void select_button_cb(Widget, XtPointer, XtPointer) { 
select_state(SELECT_TOOL); 

XmProcessTraversal(drawing_a, XmTRAVERSE_CURRENT); 

//?? XtVaSetValues(tool indicator, XmNvalue, "Select Tool", NULL); 
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static void types_tool_ok_pushed(Widget w, XtPointer client_data, 

XtPointer call_data) { 

Widget text_w = (Widget)client_data; 

XmAnyCallbackStruct *cbs = (XmAnyCallbackStruct *)call_data; 

char *text = XmTextGetString(text_w); 

char *org_text = graphic_list.global_types() ; 

enter_types = 0; // MY 8/5/97 

/* 

* MY 
*/ 


int ok, error_line, error_column, error_token_length; 
long 1; 

XmString label; 

parse_type_spec(text, &ok, &error_line, 

&error_coluinn, &error_token_length) ; 

if (ok) 

printf("parser return ok\n"); 
else 
{ 

printf("parser return NOT ok\n"); 
printf("error line = %d\n", error_line); 
printf("error column = %d\n", error_column); 
printf("error length = %d\n", error_token_length); 
putchar(007); putchar(007); putchar(007); 

1 = XmTextGetCursorPosition(text_w) ; 

1 = (long) error_line - IL; 

XmTextSetSelection(text_w, 1, 1+(long)error_token_length, 

(Time)10); 

XmTextSetCursorPosition<text_w, 1); 
return; 

} 


/* 

* MY 
*/ 

if (strcmp(text, org^text) != 0) { 

graphic_list.global_types(text); 
save state(SAVE_REQUIRED); 


free(text); text = NULL; 

free(org_text); org_text = NULL; 

XtDestroyWidget(XtParent(XtParent(XtParent(w)))); 

clear status (); 
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void types_button_cb(Widget w, XtPointer client_data, 

XtPointer call_data) { 

Widget dialog, pane, rc, text_w, action_a; 

XmString string; 
char ^description; 

static ActionArealtem action_items[] = { 

{"OK", types_tool_ok_pushed, NULL }, 

("Cancel", types_close_dialog, NULL }, 

("Help", help_cb, "types_tool.hip" } 

}; 


// MY 8/5/97 

if ( enter_types == 1 ) { 

putchar(007); 
return; 

} 

if ( enter_errs == 1 ) { 

putchar(007); 

warning(w, "Please close error message window"); 
return; 

} 

enter_types = 1; 

dialog = XtVaCreatePopupShell ("dialog", xmDialogShellWidgetClass, 

XtParent(w), 

XmNtitle, "Prototype Types Specification", 
XmNdeleteResponse, XmDESTROY, 

NULL); 

action_items[1].data = (XtPointer)dialog; //Set cancel buttons 
client_data 

pane = XtVaCreateWidget("pane", xmPanedWindowWidgetClass, dialog, 

XmNsashWidth, 1, 

XmNsashHeight, 1, 

NULL); 


rc = XtVaCreateWidget("control_area", xmRowColumnWidgetClass, pane, 
NULL); 

string = XmStringCreateSimple("View or Edit Prototype Types 
Specification"); 

XtVaCreateManagedWidget("label", xmLabelGadgetClass, rc, 

XmNlabelString, string, 

NULL); 

XmStringFree(string); 


description = graphic_list.global_types() ; 


int n =0; 

Arg args[10]; 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 


XmNrows, 

XmNcolumns, 
XmNscrollVertical, 
XmNscrollHorizontal, 
XmNeditMode, 

XmNeditable, 

XmNcursorPositionVisible, 
XmNwordWrap, 

XmNvalue, 


12); n++; 

70); n++; 
true); n++; 
true); n++; 

XmMULTI_LINE_EDIT); n++; 
true); n++; 
true); n++; 
true); n++; 
description); n++; 
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text_w = XmCreateScrolledText(rc, "text-field”, args, n); 

XtManageChild(text_w); 

//text_w = XtVaCreateManagedWidget("text-field", 
xmTextFieldWidgetClass, 

// rc, NULL); 

// XtAddCallback(text_w, XmNmodifyVerifyCallback, validate_text, 
NULL); 

// Note: If you have problems with symbols in the text, 

uncomment 

// the line above. 

XtManageChild(rc); 

//Set client data for the "OK" and "Cancel" buttons 

action_items[0].data = (XtPointer)text_w; 

action_a = CreateActionArea(pane, action_items, 

XtNumber(action^items)); 

//XtAddCallback(text_w, XmNactivateCallback, activate_cb, action_a); 

XtManageChild(pane); 


/* 

^ MY 
*/ 


{ 

GRAPH_DESC tmp_gd = gdnode; 
char buffer[100]; 
char type_name[100]; 
char type_buffer[5000]; 

strcpy(type_buffer, "") ; 

if (strcmp(XmTextGetString(text_w) , "") == 0 I | 

strcmp(XmTextGetString(text_w), "\n") == 0 ) 

while ( tmp_gd->stream_list != NULL ) 

{ 

sprintf(type_name, "#%s#", tmp_gd->stream_list->st- 
>stream_type_name) ; 

if ( strstr(type_buffer, type_name) == NULL && strcmp(type_name, 
"##") ) 

{ 

sprintf(buffer, "TYPE %s\nSPECIFICATION\nEND\nIMPLEMENTATION ADA 
%s END\n\n", 

tmp_gd->stream_list->st->stream_type_name, 
tmp_gd->stream_list->st->stream_type_name); 

XmTextInsert (text_w, OL, buffer); 
strcat(type_buffer, "#" ) ; 

strcat (type_buffer, tmp_gd->stream_list->st->stream_type_name) ; 
strcat(type_buffer, "#") ; 

} 

tmp_gd->stream_list = tmp_gd->stream_list->next ; 

} 

} 

/* 

★ my 
*/ 

free(description); 
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XtPopup(dialog, XtGrabNone); 


static void spec_tool_ok_pushed(Widget w, XtPointer client_data, 

XtPointer call_data) { 

Widget text_w = (Widget)client_data; 

XmAnyCallbackStruct *cbs = (XmAnyCallbackStruct *)call_data; 

char *text = XmTextGetString(text_w); 

char *org_text = graphic_list.cur_op_spec(); 

enter_spec = 0; // MY 8/5/97 

/* 

* MY 
*/ 


{ 


int ok, error_line, error_column, error_token_length; 
long 1; 

XmString label; 

parse_oper_spec(text, &ok, &error_line, 

&error_column,&error_token_length); 

if (ok) 

printf("parser return ok\n"); 
else 
{ 

printf("parser return NOT ok\n"); 
printf("error line = %d\n", error_line); 

printf("error column = %d\n", error_column); 
printf("error length = %d\n", error_token_length)/ 
putchar(007); putchar(007); putchar(007); 

1 == XmTextGetCursorPosition (text_w) ; 

1 = (long) error_line - IL; 

XmTextSetSelection(text_w, 1, 1+(long)error_token_length, 

(Time)10) ; 

XmTextSetCursorPosition(text_w, 1); 
return; 

} 


} 

/* 

★ my 
*/ 

if (strcmp(text, org_text) != 0) { 

graphic_list.cur_op_spec(text); 
save state(SAVE REQUIRED); 


free(text); text = NULL; 

free(org_text); org_text = NULL; 
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XtDestroyWidget(XtParent(XtParent(XtParent(w)))) ; 
clear status(); 


void spec_button_cb(Widget w, XtPointer client_data, 

XtPointer call_data) { 

Widget dialog, pane, rc, text_w, action_a; 

XmString string; 
char ^description; 

static ActionArealtem action_items[ ] = { 

{”0K", spec_tool_ok_pushed, NULL }, 

{"Cancel", spec_close_dialog, NULL }, 

("Help", help_cb, "spec_tool.hip" } 

}; 

// MY 8/5/97 

if ( enter_spec == 1 ) { 

putchar(007); 
return; 

} 

if ( enter_errs == 1 ) { 

putchar(007); 

warning(w, "Please close error message window"); 
return; 

} 

enter_spec = 1; 

dialog = XtVaCreatePopupShell ("dialog", xmDialogShellWidgetClass, 

XtParent(w), 

XmNtitle, "Prototype Specification", 
XmNdeleteResponse, XmDESTROY, 

NULL); 

action_items[1].data = (XtPointer)dialog; //Set cancel buttons 
client_data 

pane = XtVaCreateWidget("pane", xmPanedWindowWidgetClass, dialog, 

XmNsashWidth, 1, 

XmNsashHeight, 1, 

NULL); 


rc = XtVaCreateWidget("control_area", xmRowColumnWidgetClass, pane, 
NULL); 

string = XmStringCreateSimple("View or Edit Prototype Specification") 
XtVaCreateManagedWidget("label", xmLabelGadgetClass, rc, 

XmNlabelString, string, 

NULL); 

XmStringFree(string); 


description = graphic_list.cur_op_spec() ; 


int n = 0; 

Arg args[10]; 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 
XtSetArg(args[n] , 


XmNrows, 

XmNcolumns, 

XmNscrollVertical, 

XmNscrollHorizontal, 

XmNeditMode, 


12); n++; 

70); n++; 
true); n++; 
true); n++; 

XmMULTI LINE EDIT); n++; 
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XtSetArg{args[n], XmNeditable, true); n++; 

XtSetArg(args[n], XmNcursorPositionVisible, true); n++; 

XtSetArg(args[n], XmNwordWrap, true); n++; 

XtSetArg(args[n], XmNvalue, description); n++; 

text_w = XmCreateScrolledText(rc, "text-field", args, n); 
XtManageChild(text_w); 

//text_w = XtVaCreateManagedWidget("text-field", 
xmTextFieldWidgetClass, 

// rc, NULL); 

// XtAddCallback(text_w, XmNmodifyVerifyCallback, validate_text, 
NULL); 

// Note: If you have problems with *}* symbols in the text, 
uncomment 

// the line above. 

XtManageChild(rc); 

//Set client data for the "OK" and "Cancel" buttons 
action_items[0].data = (XtPointer)text_w; 

action_a = CreateActionArea(pane, action_items, 

XtNumber(action_items)); 

//XtAddCallback(text_w, XmNactivateCallback, activate_cb, action_a); 
XtManageChild(pane); 

/* 

* MY 

■k 

if (strcmp(XmTextGetString(text_w), "") == 0 || 

strcmp(XmTextGetString(text_w), "\n") == 0 ) 

XmTextInsert {text_w, OL, "SPECIFICATION\nEND"); 

•k 

k 

*/ 

free(description); 

XtPopup(dialog, XtGrabNone); 


// Callback function. Called when the radio buttons in the 
// properties dialog box are pushed. Called twice: once to 
// unselect old button, again to select the new one. 

void radio_box_cb(Widget, XtPointer which, 

XtPointer cbs) { 

XmToggleButtonCallbackStruct *state = 
(XmToggleButtonCallbackStruct *) cbs; 

if (state->set) { 

if ((int) which == 0) 
state_stream = false; 
else 

state_stream = true; 

} 

} 


296 


void save_indicator_cb(Widget widget, XtPointer, 

XtPointer cb_struct__ptr) { 

if (psdl_modified) 

handle_file_options(0) ; // save 


void error_indicator_cb(Widget widget, XtPointer client_data, 

XtPointer call_data) { 

/* 

* MY 7/22/97 

V 

if ((errors_present == NULL) || (!syntax_checked)) 

handle_psdl_options(3)/ // check syntax 

else { 

report_errors (errors_present, toplevel, next_action__ptr, 
&return_sde_flag, &prev_status); 

} 

/* 

* unmasked 8/6/97 
*/ 


// If graph_editor is invoked in viewer mode, this function 
// handles ClientMessage events from the syntax-directed editor. 
// Commented-out code handles data passed in a property, which 
// this version of the editor doesn’t take advantage of. 

// Used during testing, and left in for future use, if necessary. 

void event_handler(Widget widget, XtPointer, 

XEvent* in_event. Boolean*) { 

char buffer[INPUT_LINE_SIZE]; 

Display *display_ptr = XtDisplayOfObject(widget)/ 

Window window = DefaultRootWindow(display_ptr); 

// char **data/ 

// int return_count; 

// XTextProperty text_prop_return/ 

// Atom property_name; 
char message_in[30]; 


strcpy(message_in, in_event->xclient.data.b); 
if (strcmp(message_in, "GEDATAIN") == 0) { 

graphic_list.build_from_sde(gdnode); // @3 

#ifdef GE_DEBUG 

// printf(”graphic_list: after buildXn”); 

// graphic_list.summarize(); 

#endif 

// graphic_list.build_from_disk(); // @3 

graphic_list.draw() ; 

} 
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else if (strcmp(message_in, "PrintWindow”) == 0) 

{ 

if (PrintCmd. op == Snd_to_Prt) { 

if ((PrintCmd.printer != NULL) && (*PrintCmd.printer != ’\0*)) { 

sprintf(buffer, 

"xwd -frame -id %d | xpr -gray 2 -device ps | Ipr -P%s ”, 
XtWindow(toplevel), PrintCmd.printer); 

} 

else { 

sprintf(buffer, 

"xwd -frame -id %d | xpr -gray 2 -device ps | Ipr ", 
XtWindow(toplevel)); 

} 

setcursor(toplevel,True,XC_watch); 
system(buffer); 

setcursor(toplevel,True,XC_left_ptr) ; 

} 

else { 

if ((PrintCmd.file != NULL) && (*PrintCmd.file != *\0*)) { 

sprintf(buffer, 

"xwd -frame -id %d > %s ", 

XtWindow(toplevel), PrintCmd.file); 

} 

else { 

warning (drawing_a, "A file name must be suppled."); 

} 

setcursor(toplevel,True,XC_watch); 
system(buffer); 

setcursor(toplevel,True,XC_left_ptr) / 

} 

} 

else { 

#ifdef GE_DEBUG 

// cout << "Event " << message_in « endl; 

#endif 

} 

} 


void set_current_op() { 

char *cur_op_name/ 

cur_op_name = graphic_list. current_op_name () ; 
if (cur_op_name != NULL) 

XtVaSetValues(current_op_name, XmNvalue, cur_op_name, NULL)/ 
free(cur_op_name)/ 


void set_current_op_met() { 

char buffer[25] = "MET "; 

char *time; 

char *met = buffer; 

if (graphic_list. cur_op_spec_met () != UNDEFINED_TIME) { 

time = time_with_units (graphic_list. cur_op_spec_met () , 
graphic_list.cur_op_spec_met_unit()); 
strncat(met,time,20); 

XtVaSetValues (current_op__met, XmNvalue, met, NULL) ; 
free(time); 
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If rt 


NULL); 


} 

else 

XtVaSetValues(current_op_met, XmNvalue, 


void set_editor_title() { 

char title_str[63] = "PSDL Editor: \0"; 
char *title_ptr = title_str; 
char *name_ptr; 

name_ptr = graphic_list.root_op_name(); 
if (name_ptr != NULL) 

strncat(title_ptr,name_ptr, 50) ; 

XtVaSetValues(toplevel, XmNtitle, title_ptr, NULL); 
free(name_ptr)/ 


void init_motif() { 

// Simulated arguments 

// char* args[] = {"edit_graph”,"-geometry", "800x600",NULL}; 
// int signed_argc =3; 

// char** argv = args; 


char* args[] = {"edit_graph","-geometry","800x600",NULL}/ 
int Global_argc = 3/ 

char** Global_argv = args; 

char title_str[63] = "PSDL Editor: \0"; 

char *title_ptr = title_str; 

XmString tmp; 


print_event = (XEvent *) malloc(sizeof(XEvent)); // @7 


toplevel = 
Global_argv, 


XtVaAppInitialize(&app, "edit_graph", options, 

XtNumber(options) , &Global_argc, 

NULL, NULL); 


display_ptr = XtDisplay(toplevel); 

XtGetApplicationResources(toplevel, (XtPointer) &Resrcs, 

resources, XtNumber(resources), 
NULL, 0); 

screen__ptr = XtScreen (toplevel) ; 
initialize_color_table(screen_ptr); 
root_window = RootWindowOfScreen(screen_ptr)/ 
gcvl.foreground = BlackPixelOfScreen(screen_ptr); 
gcvl.background = WhitePixelOfScreen(screen_ptr); 
gcv2.foreground = BlackPixelOfScreen(screen_ptr); 
gcv2.background = WhitePixelOfScreen(screen_ptr); 
gcv3.foreground = WhitePixelOfScreen(screen_ptr); 
gcv3.background = WhitePixelOfScreen(screen_ptr); 
gc_mask = GCForeground 1 GCBackground; 
std_graphics_context = XCreateGC(display_ptr, 

root_window, gc_mask, &gcvl); 
dotted_context = XCreateGC(display_ptr, 

root_window, gc_mask, &gcv2); 

erase_context = XCreateGC(display_ptr, root_window, gc_mask, 

&gcv3); 
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XSetLineAttributes{display_ptr, dotted_context, 1, 

LineOnOffDash, CapButt, JoinMiter); 
XSetFunction{display_ptr, dotted_context, GXxor); 

main_w = XtVaCreateManagedWidget("main_w", xmFormWidgetClass, 

toplevel, NULL); 

build_menu_bar{main_w, menubar); 

XtManageChild(menubar); 
rowcol = 

XtVaCreateManagedWidget{"rowcol”, xmRowColumnWidgetClass, 

main_w, 

XmNnumColumns, 1, 

// XmNorientation, XmHORIZONTAL, 

NULL); 


make_buttons(rowcol, 

op_button, term_button, stream_button, select_button, 
types_button, spec_button, 
timers_button, informal_button, 
op_button_pixmap, term_button_pixmap, 
stream_button_pixmap, select_button_pixmap, 
types_button_pixmap, spec_button_pixmap, 
informal_button_pixmap, timers_button_pixmap, 
display_ptr, screen_ptr); 


XtAddCallback(op_button, 

NULL); 

XtAddCallback(term_button, 
NULL); 

XtAddCallback(stream_button, 
NULL)/ 

XtAddCallback(select_button, 
NULL); 

XtAddCallback(types_button, 
NULL); 

XtAddCallback(spec_button, 
NULL); 

XtAddCallback(timers_button, 
NULL); 

XtAddCallback(informal_button, 
informal button cb,NULL); 


XmNactivateCallback, 

XmNactivateCallback, 

XmNactivateCallback, 

XmNactivateCallback, 

XmNactivateCallback, 

XmNactivateCallback, 

XmNactivateCallback, 

XmNactivateCallback, 


op_button_cb, 
term_button_cb, 
stream_button_cb, 
select_button_cb, 
t ype s_bu11 on_cb, 
spec_button_cb, 
timers button cb. 


XtVaSetValues(toplevel, XmNtitle, title_ptr, NULL); 


current_op_name = 

XtVaCreateManagedWidget("current_op_name", xmTextWidgetClass, 

main_w, 

XmNvalue, 

XmNshadowThickness, 1, 

NULL); 


current_op_met = 

XtVaCreateManagedWidget(”current_op_met”, xmTextWidgetClass, 

main_w, 

XmNvalue, 

XmNwidth, 150, 

XmNshadowThickness, 1, 

NULL); 


scrolled_win = 

XtVaCreateManagedWidget(”scrolled_win”, 
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xmScrolledWindowWidgetClass, 

main_w, 

// XmNwidth, 1200, 

// XmNheight, 750, 

XmNscrollingPolicy, XmAUTOMATIC, 
XmNscrollBarDisplayPolicy, XmAS_NEEDED, 
NULL); 

actions.string = "draw"; 
actions.proc = draw; 

XtAppAddActions(app, &actions, 1); 
status_indicator = 

XtVaCreateManagedWidget("satus_indicator", xmTextWidgetClass, 

inain_w, 

XmNheight, 31, 

XmNvalue, 

NULL); 


save_indicator = 

XtVaCreateManagedWidget{"save_indicator", 

xmDrawnButtonWidgetClass, 
main_w, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 

XmN s ha dowT yp e, XmS HADOW_IN, 
XmNwidth, 120, 

XmNheight, 31, 

XmNmarginBottom, 13, 
XmNlabelType, XmSTRING, 

NULL); 

XtAddCallback(save_indicator, XmNactivateCallback, 
save_indicator_cb,NULL); 

error_indicator = 

XtVaCreateManagedWidget("error_indicator", 

XmDrawnButtonWidgetClass, 

main_w, 

XmNrecomputeSize, false, 
XmNpushButtonEnabled, false, 
XmNshadowType, XmSHADOW_IN, 
XmNwidth, 120, 

XmNheight, 31, 

XmNmarginBottom, 13, 
XmNlabelType, XmSTRING, 

NULL); 

XtAddCallback(error_indicator, XmNactivateCallback, 
error indicator cb, NULL); 


drawing_a = 

XtVaCreateManagedWidget{"drawing_a", 

xmDrawingAreaWidgetClass, scrolled_win, 
XmNunitType, Xml000TH_INCHES, 

XmNwidth, 11000, 

XmNheight, 8500, 

XmNresizePolicy, XmNONE, 

NULL); 

XtAddCallback(drawing_a, XmNexposeCallback, redraw, NULL); 

XtVaSetValues(drawing_a, XmNunitType, XmPIXELS, NULL); 

XmProcessTraversal(drawing_a, XmTRAVERSE_CURRENT); 

XtVaGetValues(drawing_a, XmNwidth, &width, XmNheight, &height, 
NULL); 
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drawing_area_pixmap = XCreatePixmap (display_ptr, 

root_window, width, height, 
DefaultDepthOfScreen(screen_ptr)) 
XFillRectangle(display_ptr, drawing_area_pixmap, 

erase_context, 0, 0, width, height); 

XtVaSetValues(drawing_a, XmNtranslations, 

XtParseTranslationTable(translations), NULL); 


XtVaSetValues(rowcol, 

XmNtopAttachment, 
XmNrightAttachment, 
XmNleftAttachment, 
XmNbottomAttachment, 
XmNbo11 omWidget, 
NULL); 


XmATTACH_FORM, 
XitiATTACH_NONE, 
XinATTACH_FORM, 
XmATTACH_WIDGET, 
save indicator. 


XtVaSetValues(menubar, 

XmNtopAttachment, 
XmNrightAttachment, 
XmNleftAtta chment, 
XmNleftWidget, 
XmNbottomAttachment, 
NULL); 


XmATTACH_FORM, 
XmATTACH_FORM, 
XmATTACH_WIDGET, 
rowcol, 

XmATTACH NONE, 


XtVaSetValues (current_op_name, 
XmNtopAttachment, 
XmNtopWidget, 
XmNrightAttachment, 
XmNrightWidget, 
XmNleftAttachment, 
XmNleftWidget, 
XmNbottomAttachment, 
NULL); 


XmATTACH_WIDGET, 
menubar, 

XmATTACH_WIDGET, 
current_op_met, 
XmATTACH_WIDGET, 
rowcol, 

XmATTACH NONE, 


XtVaSetValues(current_op_met, 
XmNtopAttachment, 
XmNtopWidget, 
XmNrightAttachment, 
XmNleftAttachment, 
XmNbottomAttachment, 
NULL); 


XmATTACH_WIDGET, 

menubar, 

XmATTACH_FORM, 

XmATTACH_NONE, 

XmATTACH_NONE, 


XtVaSetValues(scrolled_win, 
XmNtopAttachment, 
XmNtopWidget, 
XmNrightAttachment, 
XmNleftAttachment, 
XmNleftWidget, 
XmNbottomAttachment, 
XmNbottomWidget, 
NULL); 


XmATTACH_WIDGET, 
current_op_name, 
XmATTACH_FORM, 
XmATTACH_WIDGET, 
rowcol, 

XmATTACH_WIDGET, 
status indicator, 


XtVaSetValues(save_indicator, 
XmNtopAttachment, 
XmNrightAttachment, 
XmNleftAttachment, 
XmNbottomAttachment, 
NULL); 


XmATTACH_NONE, 
XmATTACH_NONE, 
XmATTACH_FORM, 
XmATTACH FORM, 


XtVaSetValues(error indicator. 
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XmNtopAttachment, 
XmNrightAttachment, 
XmNleftAttachment, 
XmNleftWidget, 
XmNbottomAttachment, 
NULL); 


XmATTACH_NONE, 
XmATTACH_NONE, 
XmATTACH_WIDGET, 
save_indicator, 
XmATTACH_FORM, 


XtVaSetValues(status_indicator, 
XmNtopAttachment, 
XmNrightAttachment, 
XmNleftAttachment, 
XmNleftWidget, 
XmNbottomAttachment, 


XmATTACH_NONE, 
XmATTACH_FORM, 
XmATTACH_WIDGET, 
error_indicator, 
XmATTACH_FORM, 


NULL); 


XtRealizeWidget(toplevel); 
draw_window = XtWindow(drawing_a); 

graphic_list.set_draw_environ(display_ptr, 

std_graphics_context, 
erase_context, dotted_context, 
draw_window, 

&drawing_area_pixmap, 

color_table, 

width, height); 

graphic_list. set_error__tgt (drawing_a) ; 

set_current_op(); 
set_current_op_met(); 

XmProcessTraversal(drawing_a, XmTRAVERSE_CURRENT); 

toplevel_window = XtWindowOfObject(toplevel)/ 

Atom display_id_atom = XInternAtom(display_ptr, ”WINDOW_ID”, 

False); 

XChangeProperty(display_ptr, root_window, display_id_atom, 

XA_WINDOW, 32, PropModeReplace, 

(unsigned char *) &toplevel_window, 1); 

XtAddEventHandler (toplevel, NoEventMask, true, event_handler, 

NULL); 

motif_initialized = true; 

} 

// translations provides the mappings for the keyboard 
// mapping table that allow the drawing canvas to capture 
// mouse and keyboard events. 

// The primary function, edit_graph. Modified from original main() by 
// Doug Lange 9/9/96 


* this method is added to support the edit graph and sde change over. 

* now the edit graph module is not a standalone but a method called 

* from the sde 

//extern "C" { 
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/* 

* modified 7/12/97 

* int edit_graph(...) -> void edit_graph{...) 

*/ 

void edit_graph(GRAPH_DESC current_graph, ACTION next_action, 

ERROR_MSGS sde_error_msgs) { //@2 

XEvent event; // added for custom main loop 

int reply; 

Quest^Script delete_script = 

"Deleted operators will be purged?", "Ok", "No", "Cancel", 

BTNl}; 

next_action_ptr = next_action; 

errors_present = sde_error_msgs; 

return_sde_flag = false; 

gdnode = current_graph; 

// motif_initialized assumed to be false at start of procedure 
if (motif_initialized) { 

XFillRectangle(display_ptr, drawing_area_pixmap, 
erase_context, 0, 0, width, height); 

XFillRectangle(display_ptr, draw_window, 

erase_context, 0, 0, width, height); 
if (gdnode) { 

graphic_list.build_from_sde(gdnode); 

} 

if (save_performed) 

save_state(NOT_MODIFIED) ; 
if (prev_status) { 

update_status(prev_status, false); 
free(prev_status); 
prev_status = NULL; 

} 

graphic_list.draw(); 

setcursor(toplevel,True,XC_left_ptr); 

} 

else { 

init_motif(); 

prev_status = NULL; 

save_state(NOT_MODIFIED) ; 

save^performed = false; 

default_color = WHITE; 

default_font = COURIERBOLD12; 

graphic_list.set_default_font(default_font); 

if (gdnode) { 

graphic_list.build_from_sde(gdnode); 

} 

graphic_list.draw(); 

select_state(SELECT_TOOL); 

// Initialize printer command 
PrintCmd.op = Snd_to_Prt; 

PrintCmd.printer = dup_str(""); 
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PrintCmd. file == dup_str ( ”” ) ; 

PrintCmd.answer = 0; 

// and event 

print_event->type = ClientMessage ; 
print_event->xclient.window = toplevel_window; 
print_event->xclient.format = 8; 

strcpy(print_event->xclient.data.b, "PrintWindow"); 

} 

set_editor_title()/ 
set_current_op(); 
set_current_op_met(); 

syntax_checked = true; // syntax is checked on each entry to editor 
error_label()/ 

if (graphic_list.cur_op_is_terminator()) 

XtVaSetValues(op_button, XmNsensitive, False^ NULL); 
else 

XtVaSetValues(op_button, XmNsensitive, True, NULL); 

// MY // printf(”\n”); //flushes the event queue 
XFlush(display_ptr); 

#ifdef GE_DEBUG 

cout << "Starting Motif event loop" << endl; 

#endif 

selected_object_ptr = NULL; 

// Custom main loop to check for return to sde 
do { 

XtAppNextEvent(app, &event); 

XtDispatchEvent(&event); 

if (return_sde_flag) { 

if (graphic_list.has_deleted()) { 

reply = AskUser(app, drawing_a, delete_script); 
if (reply != YES) 

return_sde_flag = false; 

} 

} 

} while (return_sde_flag == false); 

if ((next_action->option != REVERT) && 

(next_action->option != ABANDON)) 
graphic_list.write_to_sde(gdnode); 

if ((next_action_ptr->option == SAVE_TO_DISK) || 

(next_action_ptr“>option == REVERT)) // not really saved, but not 

save_performed = true; // modified 

else 

save_performed = false; // assume need to save for 

abandon 

// If we are not coming back, kill the window 
if (!next_action_ptr“>reinvoke) { 

XtUnrealizeWidget(toplevel); 

XFlush(display_ptr) ; 

} 
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else { // otherwise, will be returning, erase window and exit 

setcursor(toplevel,True,XC_watch); 

} 

return; 


//} // extern 

* - end of graph_editor.C 
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